Android 直播播放器+彈幕使用總結(jié)

本篇文章已授權(quán)微信公眾號 guolin_blog (郭霖)獨家發(fā)布

又來寫文章了,懶癌晚期拖啊拖總抽出點時間來弄屡,直播算是現(xiàn)在比較火了题禀,公司最近也要開發(fā)直播的功能。在這里分享下開發(fā)過程遇到的一些問題以及解決方案膀捷。

項目地址https://github.com/Hemumu/HLiveDemo/tree/master

筆誤迈嘹,JieCaoVideoPlayer是基于MediaPlayer的寫的,不是基于ijkplayer封裝的全庸,在此修正

現(xiàn)在有很多的開源播放器秀仲,本文所選的是基于MediaPlayer封裝的開源播放器JieCaoVideoPlayer,彈幕使用的也是B站的開源項目https://github.com/Bilibili/DanmakuFlameMaster

JieCaoVideoPlayer默認提供了基本的UI界面壶笼,但是肯定滿足不了每個人的界面要求神僵,所以我們就需要在JieCaoVideoPlayer上簡單的封裝一下。首先新建一個類繼承JCVideoPlayerStandard


public class HVideoPlayer extends JCVideoPlayerStandard {

    @Override
    public void init(final Context context) {
        super.init(context);
        this.mContext = context;
        mEditText = (EditText) findViewById(R.id.msg_edittext);
        mSendImage = (ImageView) findViewById(R.id.send_img);
        mRewardBtn = (ImageView) findViewById(R.id.reward_img);

        mSendImage.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                String content = mEditText.getText().toString();
                mSendListener.sendMsg(content);
            }
        });
        mRewardBtn.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                mShowPay.showPay();
            }
        });
    }

    @Override
    public int getLayoutId() {
        return R.layout.custom_video_player;
    }
}

JCVideoPlayerStandard對一些基本的界面操作以及頁面邏輯做了封裝覆劈,我們只需要繼承這個類保礼,然后自定義自己的布局沛励。如果有你不需要的控件就隱藏,刪除可能會報錯炮障。重寫init方法初始化一些你自定義的控件和按鈕的點擊事件目派。

JieCaoVideoPlayer是通過setUp方法來初始化播放器參數(shù),所以我們也需要來重寫這個方法來初始化我們自己的一些參數(shù)

    @Override
    public void setUp(String url, int screen, Object... objects) {
        //強制全屏
        FULLSCREEN_ORIENTATION = ActivityInfo.SCREEN_ORIENTATION_SENSOR_LANDSCAPE;
        //添加全屏下的參數(shù)
        if (objects.length != 1) {
            mSendListener = (OnSendMsgListener) objects[1];
            mShowPay = (OnPayListener) objects[2];
            mOnFullScreenListener = (OnFullScreenListener) objects[3];
        }
        super.setUp(url, screen, objects[0], mSendListener, mShowPay, mOnFullScreenListener);
        //全屏下展示彈幕
        if (currentScreen == SCREEN_WINDOW_FULLSCREEN) {
            initDanmu();
            mEditText.setVisibility(View.VISIBLE);
            mSendImage.setVisibility(View.VISIBLE);
            mOnFullScreenListener.onFullScreen(this);
        } else if (currentScreen == SCREEN_LAYOUT_NORMAL
                || currentScreen == SCREEN_LAYOUT_LIST) {
            mEditText.setVisibility(View.INVISIBLE);
            mSendImage.setVisibility(View.INVISIBLE);
        }

        //點擊返回按鈕隱藏彈幕
        backButton.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                hideDanmu();
                backPress();
            }
        });

        //重寫全屏按鈕點擊事件
        fullscreenButton.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                if (currentState == CURRENT_STATE_AUTO_COMPLETE) return;
                if (currentScreen == SCREEN_WINDOW_FULLSCREEN) {
                    hideDanmu();
                    backPress();
                } else {
                    //全屏
                    startWindowFullscreen();
                }
            }
        });
    }

需要注意一點的就是播放器器全屏胁赢,這里修改了FULLSCREEN_ORIENTATION 參數(shù)為ActivityInfo.SCREEN_ORIENTATION_SENSOR_LANDSCAPE使播放器點擊全屏后強制全屏并且是橫屏的企蹭,默認情況點擊全屏后是豎屏的,并且根據(jù)重力感應(yīng)調(diào)整屏幕方向智末。需要注意的是使用播放器的Activity需要設(shè)置為豎屏

android:screenOrientation="portrait"

否則調(diào)用橫屏后整個Activity會整個橫屏谅摄。

需要注意播放器橫屏后會創(chuàng)建一個新的播放器實例和當前的播放器不是同一個實例,也就是說點擊全屏后會重新初始化當前類吹害,并重新調(diào)用setUp方法螟凭。那怎么拿到前面小屏模式下一些必須的參數(shù)呢?查看下JCVideoPlayerStandard全屏的源碼

      public void startWindowFullscreen() {
        Log.i(TAG, "startWindowFullscreen " + " [" + this.hashCode() + "] ");
        hideSupportActionBar(getContext());
        JCUtils.getAppCompActivity(getContext()).setRequestedOrientation(FULLSCREEN_ORIENTATION);

        ViewGroup vp = (ViewGroup) (JCUtils.scanForActivity(getContext()))//.getWindow().getDecorView();
                .findViewById(Window.ID_ANDROID_CONTENT);
        View old = vp.findViewById(FULLSCREEN_ID);
        if (old != null) {
            vp.removeView(old);
        }
        textureViewContainer.removeView(JCMediaManager.textureView);
        try {
            Constructor<JCVideoPlayer> constructor = (Constructor<JCVideoPlayer>) JCVideoPlayer.this.getClass().getConstructor(Context.class);
            JCVideoPlayer jcVideoPlayer = constructor.newInstance(getContext());
            jcVideoPlayer.setId(FULLSCREEN_ID);
            FrameLayout.LayoutParams lp = new FrameLayout.LayoutParams(
                    ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
            vp.addView(jcVideoPlayer, lp);
            jcVideoPlayer.setUp(url, JCVideoPlayerStandard.SCREEN_WINDOW_FULLSCREEN, objects);
            jcVideoPlayer.setUiWitStateAndScreen(currentState);
            jcVideoPlayer.addTextureView();
            JCVideoPlayerManager.putSecondFloor(jcVideoPlayer);
            R.anim.start_fullscreen);
            CLICK_QUIT_FULLSCREEN_TIME = System.currentTimeMillis();
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    }

可以看到在全屏的時候重新創(chuàng)建了JCVideoPlayer的實例它呀,并且調(diào)用了setUp方法傳入了url以及全屏,后面這個objects是干嘛的呢棒厘?查看源碼

   public void setUp(String url, int screen, Object... objects) {
        if (!TextUtils.isEmpty(this.url) && TextUtils.equals(this.url, url)) {
            return;
        }
        this.url = url;
        this.objects = objects;
        this.currentScreen = screen;
}

可以看到這個objects是在父類的setUp中賦值的纵穿,說明我們在調(diào)setUp傳入的objects會相應(yīng)的傳入全屏播放器實例中,這也就有了上面的代碼

   if (objects.length != 1) {
            mSendListener = (OnSendMsgListener) objects[1];
            mShowPay = (OnPayListener) objects[2];
            mOnFullScreenListener = (OnFullScreenListener) objects[3];
        }
    super.setUp(url, screen, objects[0], mSendListener, mShowPay, mOnFullScreenListener);

默認的objects的第一個參數(shù)是標題奢人,后面就可以傳遞自己的一些字段谓媒,比如我們在全屏實例中需要回調(diào)一些方法,就要將這些接口傳到全屏播放器示例中何乎,否則在全屏中使用這些字段會報空指針句惯。

setUp中如果當前是全屏那么我們需要去加載彈幕,currentScreen字段是當前的狀態(tài)支救,如果是全屏就顯示彈幕否則就隱藏彈幕相關(guān)的東西抢野。關(guān)于彈幕庫的使用可以參考郭神的文章http://blog.csdn.net/guolin_blog/article/details/51933728這里我就不再細講了

   /**
     * 初始化彈幕
     */
    private void initDanmu() {
        ViewGroup vp = (ViewGroup) (JCUtils.scanForActivity(getContext()))//.getWindow().getDecorView();
                .findViewById(Window.ID_ANDROID_CONTENT);

        LayoutParams lp = new LayoutParams(
                ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
        lp.setMargins(0, sp2px(48), 0, sp2px(48));
        danmakuView = new DanmakuView(mContext);
        vp.addView(danmakuView, lp);
        danmakuView.enableDanmakuDrawingCache(true);
        danmakuView.setCallback(new DrawHandler.Callback() {
            @Override
            public void prepared() {
                showDanmaku = true;
                danmakuView.start();
                generateSomeDanmaku();
            }

            @Override
            public void updateTimer(DanmakuTimer timer) {

            }

            @Override
            public void danmakuShown(BaseDanmaku danmaku) {

            }

            @Override
            public void drawingFinished() {

            }
        });
        danmakuContext = DanmakuContext.create();
        danmakuView.prepare(parser, danmakuContext);

    }

在當直播流異常或者的或者網(wǎng)絡(luò)異常我們需要做一些操作各墨,但JCVideoPlayer并沒有提供這方面的回調(diào)指孤。又只有發(fā)揚我們的探索精神去探索源碼了


    @Override
    public void onError(int what, int extra) {
        Log.e(TAG, "onError " + what + " - " + extra + " [" + this.hashCode() + "] ");
        if (what != 38 && what != -38) {
            setUiWitStateAndScreen(CURRENT_STATE_ERROR);
        }
    }

在流異常或者網(wǎng)絡(luò)異常會打印onError日志贬堵,所以找到了這個方法恃轩,這下就簡單了重寫這個方法就行了

   @Override
    public void onError(int what, int extra) {
        super.onError(what, extra);
        //重寫onError 視頻播放錯誤的時候隱藏彈幕

         hideDanmu();

    }

默認播放上下有一個工具欄,在3秒后會自動隱藏黎做,可是我們不需要自動隱藏可以重寫這個方法

    @Override
    public void startDismissControlViewTimer() {
        //重寫父類方法叉跛,防止自動隱藏播放器工具欄。如需要自動隱藏請刪除此方法或調(diào)用super.startDismissControlViewTimer();
    }

可以通過代碼的方式自動開始播放蒸殿,如果在播放就暫停播放

jcVideoPlayer.startButton.performClick();

默認的JieCaoVideoPlayer還支持重力感應(yīng)進入全屏筷厘,只需要在Activity中加入如下代碼

JCVideoPlayer.JCAutoFullscreenListener sensorEventListener;
SensorManager                          sensorManager;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
    sensorEventListener = new JCVideoPlayer.JCAutoFullscreenListener();
}
@Override
protected void onResume() {
    super.onResume();
    Sensor accelerometerSensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
    sensorManager.registerListener(sensorEventListener, accelerometerSensor, SensorManager.SENSOR_DELAY_NORMAL);
}
@Override
protected void onPause() {
    super.onPause();
    sensorManager.unregisterListener(sensorEventListener);
}

JieCaoVideoPlayer還支持浮層小窗播放挽铁,能在ListViewViewPagerListView敞掘、ViewPagerFragment等多重嵌套模式下全屏工作叽掘,源碼的類大部分方法都是public需要什么重寫就行了。

使用
<com.helin.hlivedemo.view.HVideoPlayer
                android:id="@+id/custom_videoplayer_standard"
                android:layout_width="match_parent"
                android:layout_height="200dp" />

Acitivity中生命周期中加入對播放器的管理

    @Override
    public void onBackPressed() {
        if (JCVideoPlayer.backPress()) {
            //隱藏彈幕
            if (mFullScreenPlayer != null) {
                mFullScreenPlayer.hideDanmu();
                mFullScreenPlayer=null;
            }
            return;
        }
        super.onBackPressed();
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mVideoPlayerStandard.danmaDes();
    }

    @Override
    protected void onPause() {
        super.onPause();
        JCVideoPlayer.releaseAllVideos();
        if (mFullScreenPlayer != null) {
            mFullScreenPlayer.hideDanmu();
        }
    }

    @Override
    protected void onResume() {
        super.onResume();
        mVideoPlayerStandard.danmaResume();
    }

還可以添加UserAction對播放器的各種狀態(tài)監(jiān)聽

mVideoPlayerStandard.setJcUserAction(new JCUserAction() {
            @Override
            public void onEvent(int type, String url, int screen, Object... objects) {
                switch (type){
                    //開始播放
                    case JCVideoPlayer.CURRENT_STATE_PLAYING:

                        break;
                     //暫停播放
                    case JCVideoPlayer.CURRENT_STATE_PAUSE:

                        break;

                }
            }
        });

最后效果如下

GIF.gif
demo中的直播流不太穩(wěn)定大家可以替換成自己覺得穩(wěn)定的直播流玖雁,或者換成一個視頻也可以更扁。有什么問題歡迎交流!
Thanks

https://github.com/Bilibili/DanmakuFlameMaster
https://github.com/Bilibili/ijkplayer
https://github.com/lipangit/JieCaoVideoPlayer

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末赫冬,一起剝皮案震驚了整個濱河市浓镜,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌劲厌,老刑警劉巖膛薛,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異补鼻,居然都是意外死亡哄啄,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進店門风范,熙熙樓的掌柜王于貴愁眉苦臉地迎上來咨跌,“玉大人,你說我怎么就攤上這事硼婿⌒堪耄” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵寇漫,是天一觀的道長刊殉。 經(jīng)常有香客問我,道長州胳,這世上最難降的妖魔是什么记焊? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮陋葡,結(jié)果婚禮上亚亲,老公的妹妹穿的比我還像新娘。我一直安慰自己腐缤,他們只是感情好捌归,可當我...
    茶點故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著岭粤,像睡著了一般惜索。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上剃浇,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天巾兆,我揣著相機與錄音猎物,去河邊找鬼。 笑死角塑,一個胖子當著我的面吹牛蔫磨,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播圃伶,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼堤如,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了窒朋?” 一聲冷哼從身側(cè)響起搀罢,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎侥猩,沒想到半個月后榔至,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡欺劳,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年唧取,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片杰标。...
    茶點故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡兵怯,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出腔剂,到底是詐尸還是另有隱情,我是刑警寧澤驼仪,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布掸犬,位于F島的核電站,受9級特大地震影響绪爸,放射性物質(zhì)發(fā)生泄漏湾碎。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一奠货、第九天 我趴在偏房一處隱蔽的房頂上張望介褥。 院中可真熱鬧,春花似錦递惋、人聲如沸柔滔。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽睛廊。三九已至,卻和暖如春杉编,著一層夾襖步出監(jiān)牢的瞬間超全,已是汗流浹背咆霜。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留嘶朱,地道東北人蛾坯。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像疏遏,于是被迫代替她去往敵國和親脉课。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,762評論 2 345

推薦閱讀更多精彩內(nèi)容