編輯:關於Android編程
前言
本章將實現非常實用的功能——下載在線視頻。涉及到多線程、線程更新UI等技術,還需思考產品的設計,如何將新加的功能更好的融入到現有的產品中,並不是簡單的加一個界面就行了,歡迎大家交流產品設計和技術細節實現!
系列
1、Android 使用Vitamio打造自己的萬能播放器(1)——准備
2、Android 使用Vitamio打造自己的萬能播放器(2)—— 手勢控制亮度、音量、縮放
3、Android 使用Vitamio打造自己的萬能播放器(3)——本地播放(主界面、視頻列表)
4、Android 使用Vitamio打造自己的萬能播放器(4)——本地播放(快捷搜索、數據存儲)
5、Android 使用Vitamio打造自己的萬能播放器(5)——在線播放(播放優酷視頻)
6、Android 使用Vitamio打造自己的萬能播放器(6)——在線播放(播放列表)
正文
一、目標
本章實現視頻下載的功能


使用說明:進入在線視頻,點擊播放時將彈出選擇框詢問播放還是下載,點擊下載後進度條將在本地視頻頂部顯示。如果想邊看便下載,請直接點擊本地播放列表中正在下載的視頻。
二、實現(部分主要實現代碼)
FileDownloadHelper
public class FileDownloadHelper {
private static final String TAG = "FileDownloadHelper";
/** 線程池 */
private ThreadPool mPool = new ThreadPool();
/** 開始下載 */
public static final int MESSAGE_START = 0;
/** 更新進度 */
public static final int MESSAGE_PROGRESS = 1;
/** 下載結束 */
public static final int MESSAGE_STOP = 2;
/** 下載出錯 */
public static final int MESSAGE_ERROR = 3;
/** 中途終止 */
private volatile boolean mIsStop = false;
private Handler mHandler;
public volatile HashMap<String, String> mDownloadUrls = new HashMap<String, String>();
public FileDownloadHelper(Handler handler) {
if (handler == null)
throw new IllegalArgumentException("handler不能為空!");
this.mHandler = handler;
}
public void stopALl() {
mIsStop = true;
mPool.stop();
}
public void newDownloadFile(final String url) {
newDownloadFile(url, Environment.getExternalStorageDirectory() + "/" + FileUtils.getUrlFileName(url));
}
/**
* 下載一個新的文件
*
* @param url
* @param savePath
*/
public void newDownloadFile(final String url, final String savePath) {
if (mDownloadUrls.containsKey(url))
return;
else
mDownloadUrls.put(url, savePath);
mPool.start(new Runnable() {
@Override
public void run() {
mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_START, url));
HttpClient client = new DefaultHttpClient();
HttpGet get = new HttpGet(url);
InputStream inputStream = null;
FileOutputStream outputStream = null;
try {
HttpResponse response = client.execute(get);
HttpEntity entity = response.getEntity();
final int size = (int) entity.getContentLength();
inputStream = entity.getContent();
if (size > 0 && inputStream != null) {
outputStream = new FileOutputStream(savePath);
int ch = -1;
byte[] buf = new byte[1024];
//每秒更新一次進度
new Timer().schedule(new TimerTask() {
@Override
public void run() {
try {
FileInputStream fis = new FileInputStream(new File(savePath));
int downloadedSize = fis.available();
if (downloadedSize >= size)
cancel();
mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_PROGRESS, downloadedSize, size, url));
} catch (Exception e) {
}
}
}, 50, 1000);
while ((ch = inputStream.read(buf)) != -1 && !mIsStop) {
outputStream.write(buf, 0, ch);
}
outputStream.flush();
}
} catch (Exception e) {
Log.e(TAG, e.getMessage(), e);
mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_ERROR, url + ":" + e.getMessage()));
} finally {
try {
if (outputStream != null)
outputStream.close();
} catch (IOException ex) {
}
try {
if (inputStream != null)
inputStream.close();
} catch (IOException ex) {
}
}
mDownloadUrls.remove(url);
mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_STOP, url));
}
});
}
}
代碼說明:
a. ThreadPool是線程池,請參照項目代碼。
b. 這裡使用了Time定時來刷進度,而沒有直接在write數據時更新進度,這樣的原因時每秒write較高,更新UI過於頻繁,可能導致超時等問題。
Handle
public Handler mDownloadHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
PFile p;
String url = msg.obj.toString();
switch (msg.what) {
case FileDownloadHelper.MESSAGE_START://開始下載
p = new PFile();
p.path = mParent.mFileDownload.mDownloadUrls.get(url);
p.title = new File(p.path).getName();
p.status = 0;
p.file_size = 0;
if (mDownloadAdapter == null) {
mDownloadAdapter = new FileAdapter(getActivity(), new ArrayList<PFile>());
mDownloadAdapter.add(p, url);
mTempListView.setAdapter(mDownloadAdapter);
mTempListView.setVisibility(View.VISIBLE);
} else {
mDownloadAdapter.add(p, url);
mDownloadAdapter.notifyDataSetChanged();
}
break;
case FileDownloadHelper.MESSAGE_PROGRESS://正在下載
p = mDownloadAdapter.getItem(url);
p.temp_file_size = msg.arg1;
p.file_size = msg.arg2;
int status = (int) ((msg.arg1 * 1.0 / msg.arg2) * 10);
if (status > 10)
status = 10;
p.status = status;
mDownloadAdapter.notifyDataSetChanged();
break;
case FileDownloadHelper.MESSAGE_STOP://下載結束
p = mDownloadAdapter.getItem(url);
FileBusiness.insertFile(getActivity(), p);
break;
case FileDownloadHelper.MESSAGE_ERROR:
Toast.makeText(getActivity(), url, Toast.LENGTH_LONG).show();
break;
}
super.handleMessage(msg);
}
};
代碼說明:
a. mTempListView是新增的,默認是隱藏,請參見項目代碼layout部分。
b. 下載流程:開始(顯示mTempListView) -> 正在下載(更新進度圖片和大小) -> 完成(入褲)
Dialog
if (FileUtils.isVideoOrAudio(url)) {
Dialog dialog = new AlertDialog.Builder(getActivity()).setIcon(android.R.drawable.btn_star).setTitle("播放/下載").setMessage(url).setPositiveButton("播放", new OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent intent = new Intent(getActivity(), VideoPlayerActivity.class);
intent.putExtra("path", url);
startActivity(intent);
}
}).setNeutralButton("下載", new OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
MainFragmentActivity activity = (MainFragmentActivity) getActivity();
activity.mFileDownload.newDownloadFile(url);
Toast.makeText(getActivity(), "正在下載 .." + FileUtils.getUrlFileName(url) + " ,可從本地視頻查看進度!", Toast.LENGTH_LONG).show();
}
}).setNegativeButton("取消", null).create();
dialog.show();
return true;
}
三、下載
至本章節往後,代碼均不再提供下載,請移步Google Code:
http://code.google.com/p/android-oplayer
以上就是對Android Vitamio 開發播放器下載視頻播放器的資料整理,有需要開發Android播放器的朋友可以參考下。
Android 網絡學習之使用多線程下載,支持斷點續傳
既然本節是學習如何使用多線程下載,那我們先要明白什麼是多線程下載,在搞明白什麼是多線程下載之前,需要先知道什麼是單線程下載。上圖就是說明了單線程下載的原來,因此單線程下載
Android Wear
原文地址:http://developer.android.com/design/wear/index.html 前言 設計Android Wear可穿戴設備應用程
領略千變萬化的Android Drawable (二)
hello,上篇我們已經分析6種Drawable的使用方法,本篇咱們就繼續剩下的Drawable~,閒話莫多說,那就直接開始吧。7、TransitionDrawable很
Android實現View滑動的幾種方式
什麼是View?實現View滑動的方式有哪些?1. 關於View我們需要知道的(1)什麼是View? Android中的View類是所