實現安卓裡邊下邊播的播放器(源碼公開)

来源:https://www.cnblogs.com/kingthy/archive/2018/02/05/xlvideoplayer.html
-Advertisement-
Play Games

前段時間弄了一款安卓電視盒子的遠程遙控輸入法APP:TVRemoteIME,此APP實現了遠程跨屏的輸入、遙控和應用管理功能。 最近發現盒子上要播放電影資源除了買APP會員之外,能直接免費播放電影的第三方APP越來越少了,要麼更新不及時要麼電影資源非常的少或者廣告繁多。而在電腦上要找一部電影播放還是... ...


一、前言:

前段時間弄了一款安卓電視盒子的遠程遙控輸入法APP:TVRemoteIME,此APP實現了遠程跨屏的輸入、遙控和應用管理功能。

最近發現盒子上要播放電影資源除了買APP會員之外,能直接免費播放電影的第三方APP越來越少了,要麼更新不及時要麼電影資源非常的少或者廣告繁多。而在電腦上要找一部電影播放還是非常容易的,因為網路上個人搭建的電影資源網站繁多或者BT下載等等,於是想到在我的TVRemoteIME上增加播放器功能,這樣在控制端(手機,電腦,PAD)直接輸入一個播放資源地址或者上傳一個電影資源文件(視頻文件或者種子文件)即可在電視盒子上播放。

有了想法,就開始行動……

 

二、下載功能的實現

現網路上的電影資源文件基本上要下載回來才可以實現播放,下載地址格式很多都是迅雷、ed2k、種子文件(磁力鏈)等方式。要實現邊下載邊播放功能,首要的就是解決資源下載的問題。最初想法是實現種子文件的下載功能,也就是實現BT協議即可。因為之前有瞭解過MonoTorrent這個開源項目,所以認為在安卓里要實現BT下載問題也應該不大。由於初入安卓之門,於是想找找有沒有可利用的現有“輪子”,在GitHub搜索時,卻意外的發現了這個MiniThunder項目,它已完全實現了種子、ed2k、thunder等協議的文件下載功能,並且還支持視頻的邊下載邊播放功能!完全就是我想要的東西!

具體使用方法的示例代碼:

//初始化
XLTaskHelper.init(context);

//添加網路文件的下載任務(http://, thunder://, ed2k://, ftp:// 等協議)
XLTaskHelper.instance().addThunderTask(url, localSavePath, null);

//添加種子文件的下載任務
XLTaskHelper.instance().addTorrentTask(filename, localSavePath, indexs);

//獲取視頻文件的本地播放地址(要求任務正在下載)
XLTaskHelper.instance().getLoclUrl(this.localSavePath + item.getName());

 

註:MiniThunder項目是利用迅雷庫實現的功能,具體使用許可就暫時不明瞭,建議勿用於商業用途。測試過程中發現磁力鏈在項目庫是有可添加下載任務,但卻是無法下載,應該是迅雷已關閉了下載介面。

 

三、播放器的實現

安卓里的播放器現有的開源與不開源的項目太多了,比如安卓原生的VideoView或者Google的ExoPlayer項目,國內的有B站的ijkplayer,百度的播放器SDK,迅雷的Aplayer播放器引擎等等。原生的VideoView支持的視頻格式太少了所以第一個放棄使用。最後選擇了B站的ijkplayer,因為完全開源並且支持的視頻協議非常的多。在Github能搜索到非常多的ijkplayer播放器示例項目代碼,直接使用現有的“輪子”能省去自己設計UI界面的麻煩,於是找到了一個AFAP Player項目,裡面已做好了百度和ijkplayer的示例播放器,界面非常的簡潔,非常的適合我的要求。但為了能實現播放列表的功能,在AFAP Player的基礎上我還做了一些功能增加,且由於播放器是要在電視盒子上播放,無法進行手觸摸控制,所以需要做遙控器控制的相容處理。

針對遙控器的操作我們主要實現以下功能:

1、按左右鍵實現播放的快退、快進功能

2、按上下鍵實現播放列表的選擇(如視頻源有多個的情況,比如種子資源文件里可能會包含非常多的視頻文件)

3、按確定鍵實現播放及暫停播放功能

4、按返回鍵退出播放器

 

功能實現代碼如下:(代碼摘錄於TVRemoteIME的XLVideoPlayActivity.java文件)

   private boolean changeProgressByKey = false;
    private int oldProgressValue = -1;
    private int newProgressValue = -1;
    @Override
    public boolean onKeyUp(int keyCode, KeyEvent event) {
        switch (keyCode) {
            case KeyEvent.KEYCODE_DPAD_LEFT:
            case KeyEvent.KEYCODE_DPAD_RIGHT:
                if(changeProgressByKey){
                    changeProgressByKey = false;
                    oldProgressValue = -1;
                    endGesture();
                }
                break;
        }
        return super.onKeyUp(keyCode, event);
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        switch (keyCode){
            case KeyEvent.KEYCODE_ESCAPE:
            case KeyEvent.KEYCODE_BACK:
                if(playListView.isShown()) {
                    show(defaultTimeout);
                    return true;
                }
                break;
            case KeyEvent.KEYCODE_DPAD_LEFT:
            case KeyEvent.KEYCODE_DPAD_RIGHT:
                if(!changeProgressByKey)changeProgressByKey = true;
                if(oldProgressValue == -1){
                    oldProgressValue = 0;
                    newProgressValue = oldProgressValue;
                }
                newProgressValue += keyCode == KeyEvent.KEYCODE_DPAD_LEFT ? -1 : 1;
                Log.d(TAG, "newProgressValue = " + newProgressValue);
                if(newProgressValue < (0 - seekBar.getMax()))newProgressValue = (0 - seekBar.getMax());
                if(newProgressValue > seekBar.getMax())newProgressValue = seekBar.getMax();
                float deltaP = oldProgressValue - newProgressValue;
                onProgressSlide(-deltaP / seekBar.getMax());
                return true;
            case KeyEvent.KEYCODE_DPAD_DOWN:
            case KeyEvent.KEYCODE_DPAD_UP:
                if(playListView.isShown()){
                    View view = playListView.getLayoutManager().getFocusedChild();
                    if(view != null){
                        View nextView = playListView.getLayoutManager().onInterceptFocusSearch(view, keyCode == KeyEvent.KEYCODE_DPAD_DOWN ? View.FOCUS_DOWN : View.FOCUS_UP);
                        if(nextView != null)nextView.requestFocus();
                    }else {
                        playListView.requestFocus(keyCode == KeyEvent.KEYCODE_DPAD_DOWN ? View.FOCUS_DOWN : View.FOCUS_UP);
                    }
                    return true;
                }else if(xlDownloadManager.taskInstance().getPlayList().size() > 1){
                    playListView.setVisibility(View.VISIBLE);
                    return true;
                }
                break;
            case KeyEvent.KEYCODE_ENTER:
            case KeyEvent.KEYCODE_DPAD_CENTER:
                doPauseResume();
                show(defaultTimeout);
                return true;
        }
        return super.onKeyDown(keyCode, event);
    }

註:由於快進或快退可能會連接跳過一段播放時間,也就是在遙控操作時會一直按住左右鍵不放。所以代碼里處理左右鍵按下事件時只記錄進度值,在左右鍵彈上事件時才執行快退/快進功能。

 

 

四、邊下邊播的功能實現

下載功能及播放器兩個“輪子”都有了,要實現邊下邊播的功能,只要將這兩個“輪子”組裝起來就好了。在這裡我寫了一個DownloadTask類來實現這功能的整合。此類的完全代碼請參考項目代碼。

1、在啟動播放器前需要接收一個視頻源地址參數:

mVideoPath = getIntent().getStringExtra("videoPath");

此視頻源地址支持直播源地址(http://, rtmp://, mms://)、本地視頻、種子文件(.torrent)、網路視頻源(thunder://, ed2k://)。

 

2、將視頻源地址傳遞給DownloadTask類處理

xlDownloadManager.taskInstance().setUrl(mVideoPath);

DownloadTask會分析此視頻源地址的視頻格式,分析出是直播源還是本地文件或者網路視頻文件,如果是種子文件還會對種子文件進行分析,只取種子文件里的視頻文件進行處理。

    public void setUrl(String url) {
        this.url = url;

        //刪除舊任務及文件
        this.stopTask();
        this.playList.clear();
        this.mIsLiveMedia = FileUtils.isLiveMedia(this.url);
        this.isNetworkDownloadTask = !this.mIsLiveMedia && FileUtils.isNetworkDownloadTask(this.url);
        this.name = this.mIsLiveMedia ? FileUtils.getWebMediaFileName(this.url) :
                     this.isNetworkDownloadTask ? XLTaskHelper.instance().getFileName(this.url) : FileUtils.getFileName(this.url);
        this.localSavePath = (new File(getBaseDir(), FileUtils.getFileNameWithoutExt(this.name)).toString()) + "/";
        this.isLocalMedia = !this.mIsLiveMedia && !this.isNetworkDownloadTask && FileUtils.isMediaFile(this.name);
        this.torrentInfo = null;
        this.torrentMediaIndexs = null;
        this.torrentUnmediaIndexs = null;
        this.currentPlayMediaIndex = 0;
        if(this.isLocalMedia){
            playList.add(new PlayListItem(this.name, 0, new File(this.getUrl()).length()));
        }else if(this.mIsLiveMedia || this.isNetworkDownloadTask){
            playList.add(new PlayListItem(this.name, 0, 0L));
        } else if (".torrent".equals(FileUtils.getFileExt(this.name))) {
            this.torrentInfo = XLTaskHelper.instance().getTorrentInfo(this.url);
            this.initTorrentIndexs();
        }
    }

3、啟動下載任務

xlDownloadManager.taskInstance().startTask()

DownloadTask啟動任務時會根據視頻源的格式做相應的處理,如果是直播源與本地視頻文件則不會做下載處理,而如果是種子文件或者網路視頻文件則會調用XLTaskHelper添加下載任務

   public boolean startTask(){
        if(TextUtils.isEmpty(this.url) || this.taskId != 0L){
            return false;
        }

        if(this.isNetworkDownloadTask){
            if(this.url.toLowerCase().startsWith("magnet:?")){
                Log.e(TAG, "暫時不支持magnet鏈的下載播放");
                return false;
            }else {
                taskId = XLTaskHelper.instance().addThunderTask(this.url, localSavePath, null);
            }
        }else if(this.torrentInfo != null) {
            if(this.currentPlayMediaIndex != -1) {
                try {
                    taskId = XLTaskHelper.instance().addTorrentTask(this.url, localSavePath, this.getTorrentDeselectedIndexs());
                } catch (Exception e) {
                }
            }
        }else {
            taskId = this.isLocalMedia || this.mIsLiveMedia ? -9999L : 0L;
        }
        Log.d(TAG, "startTask(" + this.url + "), taskId = " + taskId);
        return  taskId != 0L;
    }

 

4、開始邊下載邊播放

mVideoView.setVideoPath(xlDownloadManager.taskInstance().getPlayUrl());

DownloadTask獲取播放地址時,如果是種子文件或者網路視頻文件則獲取mini_thunder的本地播放地址,否則直接返回播放源地址

    public String getPlayUrl(){
        if(this.isLocalMedia || this.mIsLiveMedia){
            return this.getUrl();
        }else if(this.taskId != 0L){
            if(this.isNetworkDownloadTask){
                return XLTaskHelper.instance().getLoclUrl(this.localSavePath + this.name);
            }else if(this.torrentInfo != null && this.currentPlayMediaIndex != -1){
                for(PlayListItem item : getPlayList()){
                    if(item.getIndex() == this.currentPlayMediaIndex){
                        return XLTaskHelper.instance().getLoclUrl(this.localSavePath + item.getName());
                    }
                }
            }
        }
        return null;
    }

 

五、播放器的調用方法

播放器封裝好後,外部要調用視頻播放時一行代碼即可實現播放功能:

XLVideoPlayActivity.intentTo(context, url, title);

url參數即是可支持的直播源、本地文件、種子文件或者網路視頻文件地址。

要查看播放效果請參考 TVRemoteIME APP(TV盒子安裝)。

六、結束

項目開源地址:TVRemoteIME

註:由於此播放器屬於TVRemoteIME項目下的子模塊項目,所以項目代碼寄生於它,但目前TVRemoteIME的代碼暫時不開源,後期視情況再決定是否開源。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 大多數Windows用戶,右鍵中預設新建的文件格式就那麼幾種。這可能對我們愛折騰或者特殊要求的人員來說可能有點不方便,比如說要創建個 .reg .bat .vbs等格式的文件,就需要新建個文本文件修改尾碼為或者是通過對應軟體保存為 .reg .bat .vbs等。是不是有點麻煩喲,可不可以在右鍵中的 ...
  • 索引: 使用索引可快速訪問資料庫表中的特定信息。索引是對資料庫表中一列或多列的值進行排序的一種結構,例如 employee 表的姓(name)列。如果要按姓查找特定職員,與必須搜索表中的所有行相比,索引會幫助您更快地獲得該信息。 索引是一個單獨的、物理的資料庫結構,它是某個表中一列或若幹列值的集合和 ...
  • https://github.com/mono/old-code https://wiki.scn.sap.com/wiki/display/SQLANY/SQL+Anywhere+and+Microsoft+.NET http://www.mono-project.com/docs/databas ...
  • DML和DQL 增刪改查 SELECT * FROM grade --新增 insert -- 向年級表中新增3條數據INSERT INTO grade(gradeID,gradeName) VALUES(4,'4年級');INSERT INTO grade(gradeID,gradeName) V ...
  • 一,iOS設備中的蜂窩網路通信棧運行在專門的晶元上,這個晶元就是數字基帶處理器。 參考資料:《黑客攻防技術寶典-iOS實戰篇》 ...
  • 尊重勞動成果,轉載請標明出處:http://www.cnblogs.com/tangZH/p/8419053.html 在做項目的過程中,遇到了一個奇怪的現象,我設置RelativeLayout為的寬度為wrap_content,而且RelativeLayout裡面的組件也設置了固定大小,可是Rel ...
  • 尊重勞動成果,轉載請標明出處:http://www.cnblogs.com/tangZH/p/8419010.html 我們在項目中經常需在一個listview中展示不一樣的佈局,我們可以在adapter的getView()中根據position來決定該展示哪些佈局。 我在項目中便是如此,第一個it ...
  • 前言:圖片選擇器基本上是每個App必備的東西,用公認好的第三方也可以,但是自己寫的改起來方便,用起來順手,而且這東西想想可能沒動手之前想想比較難,實際操作起來就很簡單了,這次先主要寫流程,具體優化的細節以後在寫。 難點:動手之前最困惑的問題就是怎麼獲取到手機里所有的圖片,獲取到之後,顯示出來,處理邏 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...