前段時間弄了一款安卓電視盒子的遠程遙控輸入法APP:TVRemoteIME,此APP實現了遠程跨屏的輸入、遙控和應用管理功能。html
最近發現盒子上要播放電影資源除了買APP會員以外,能直接免費播放電影的第三方APP愈來愈少了,要麼更新不及時要麼電影資源很是的少或者廣告繁多。而在電腦上要找一部電影播放仍是很是容易的,由於網絡上我的搭建的電影資源網站繁多或者BT下載等等,因而想到在個人TVRemoteIME上增長播放器功能,這樣在控制端(手機,電腦,PAD)直接輸入一個播放資源地址或者上傳一個電影資源文件(視頻文件或者種子文件)便可在電視盒子上播放。java
有了想法,就開始行動……git
現網絡上的電影資源文件基本上要下載回來才能夠實現播放,下載地址格式不少都是迅雷、ed2k、種子文件(磁力鏈)等方式。要實現邊下載邊播放功能,首要的就是解決資源下載的問題。最初想法是實現種子文件的下載功能,也就是實現BT協議便可。由於以前有了解過MonoTorrent這個開源項目,因此認爲在安卓裏要實現BT下載問題也應該不大。因爲初入安卓之門,因而想找找有沒有可利用的現有「輪子」,在GitHub搜索時,卻意外的發現了這個MiniThunder項目,它已徹底實現了種子、ed2k、thunder等協議的文件下載功能,而且還支持視頻的邊下載邊播放功能!徹底就是我想要的東西!github
具體使用方法的示例代碼:網絡
//初始化 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項目是利用迅雷庫實現的功能,具體使用許可就暫時不明瞭,建議勿用於商業用途。測試過程當中發現磁力鏈在項目庫是有可添加下載任務,但倒是沒法下載,應該是迅雷已關閉了下載接口。ide
安卓裏的播放器現有的開源與不開源的項目太多了,好比安卓原生的VideoView或者Google的ExoPlayer項目,國內的有B站的ijkplayer,百度的播放器SDK,迅雷的Aplayer播放器引擎等等。原生的VideoView支持的視頻格式太少了因此第一個放棄使用。最後選擇了B站的ijkplayer,由於徹底開源而且支持的視頻協議很是的多。在Github能搜索到很是多的ijkplayer播放器示例項目代碼,直接使用現有的「輪子」能省去本身設計UI界面的麻煩,因而找到了一個AFAP Player項目,裏面已作好了百度和ijkplayer的示例播放器,界面很是的簡潔,很是的適合個人要求。但爲了能實現播放列表的功能,在AFAP Player的基礎上我還作了一些功能增長,且因爲播放器是要在電視盒子上播放,沒法進行手觸摸控制,因此須要作遙控器控制的兼容處理。測試
針對遙控器的操做咱們主要實現如下功能:網站
一、按左右鍵實現播放的快退、快進功能this
二、按上下鍵實現播放列表的選擇(如視頻源有多個的狀況,好比種子資源文件裏可能會包含很是多的視頻文件)url
三、按肯定鍵實現播放及暫停播放功能
四、按返回鍵退出播放器
功能實現代碼以下:(代碼摘錄於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類來實現這功能的整合。此類的徹底代碼請參考項目代碼。
一、在啓動播放器前須要接收一個視頻源地址參數:
mVideoPath = getIntent().getStringExtra("videoPath");
此視頻源地址支持直播源地址(http://, rtmp://, mms://)、本地視頻、種子文件(.torrent)、網絡視頻源(thunder://, ed2k://)。
二、將視頻源地址傳遞給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(); } }
三、啓動下載任務
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; }
四、開始邊下載邊播放
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的代碼暫時不開源,後期視狀況再決定是否開源。