WebView中实现文件下载功能

WebView控制调用相应的WEB页面进行展示。当碰到页面有下载链接的时候,点击上去是一点反应都没有的。原来是因为WebView默认没有开启文件下载的功能,如果要实现文件下载的功能,需要设置WebView的DownloadListener,通过实现自己的DownloadListener来实现文件的下载。具体操作如下:

1、设置WebView的DownloadListener: 
    webView.setDownloadListener(new MyWebViewDownLoadListener());

2、实现MyWebViewDownLoadListener这个类,具体可以如下这样:

private class MyWebViewDownLoadListener implements DownloadListener{

@Override

public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype,

long contentLength) {

Log.i("tag", "url="+url);

Log.i("tag", "userAgent="+userAgent);

Log.i("tag", "contentDisposition="+contentDisposition);

Log.i("tag", "mimetype="+mimetype);

Log.i("tag", "contentLength="+contentLength);

Uri uri = Uri.parse(url);

Intent intent = new Intent(Intent.ACTION_VIEW, uri);

startActivity(intent);

}

}


private class MyWebViewDownLoadListener implements DownloadListener{

@Override

public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype,

long contentLength) {

Log.i("tag", "url="+url);

Log.i("tag", "userAgent="+userAgent);

Log.i("tag", "contentDisposition="+contentDisposition);

Log.i("tag", "mimetype="+mimetype);

Log.i("tag", "contentLength="+contentLength);

Uri uri = Uri.parse(url);

Intent intent = new Intent(Intent.ACTION_VIEW, uri);

startActivity(intent);

}

}

这只是调用系统中已经内置的浏览器进行下载,还没有WebView本身进行的文件下载,不过,这也基本上满足我们的应用场景了。

我在项目中的运用 
项目要求这样: 
1,需要使用WebView加载一个网页; 
2,网页中有文件下载的链接,点击后需要下载文件到SDcard; 
3,然后自动打开文件; 
下面是具体解决办法 
第一步,对WebView进行一系列设置。

WebView webview=(WebView)layout.findViewById(R.id.webview);

webview.getSettings().setJavaScriptEnabled(true);

webview.setWebChromeClient(new MyWebChromeClient());

webview.requestFocus();

//              webview.loadUrl("file:///android_asset/risktest.html");

webview.loadUrl(jcrs_sub.get(position).addr);

// 设置web视图客户端

webview.setWebViewClient(new MyWebViewClient());

webview.setDownloadListener(new MyWebViewDownLoadListener());

//内部类

public class MyWebViewClient extends WebViewClient {

// 如果页面中链接,如果希望点击链接继续在当前browser中响应,

// 而不是新开Android的系统browser中响应该链接,必须覆盖 webview的WebViewClient对象。

public boolean shouldOverviewUrlLoading(WebView view, String url) {

L.i("shouldOverviewUrlLoading");

view.loadUrl(url);

return true;

}

public void onPageStarted(WebView view, String url, Bitmap favicon) {

L.i("onPageStarted");

showProgress();

}

public void onPageFinished(WebView view, String url) {

L.i("onPageFinished");

closeProgress();

}

public void onReceivedError(WebView view, int errorCode,

String description, String failingUrl) {

L.i("onReceivedError");

closeProgress();

}

}

// 如果不做任何处理,浏览网页,点击系统“Back”键,整个Browser会调用finish()而结束自身,

// 如果希望浏览的网 页回退而不是推出浏览器,需要在当前Activity中处理并消费掉该Back事件。

public boolean onKeyDown(int keyCode, KeyEvent event) {

// if((keyCode==KeyEvent.KEYCODE_BACK)&&webview.canGoBack()){

// webview.goBack();

// return true;

// }

return false;

}


WebView webview=(WebView)layout.findViewById(R.id.webview);

webview.getSettings().setJavaScriptEnabled(true);

webview.setWebChromeClient(new MyWebChromeClient());

webview.requestFocus();

//              webview.loadUrl("file:///android_asset/risktest.html");

webview.loadUrl(jcrs_sub.get(position).addr);

// 设置web视图客户端

webview.setWebViewClient(new MyWebViewClient());

webview.setDownloadListener(new MyWebViewDownLoadListener());

//内部类

public class MyWebViewClient extends WebViewClient {

// 如果页面中链接,如果希望点击链接继续在当前browser中响应,

// 而不是新开Android的系统browser中响应该链接,必须覆盖 webview的WebViewClient对象。

public boolean shouldOverviewUrlLoading(WebView view, String url) {

L.i("shouldOverviewUrlLoading");

view.loadUrl(url);

return true;

}

public void onPageStarted(WebView view, String url, Bitmap favicon) {

L.i("onPageStarted");

showProgress();

}

public void onPageFinished(WebView view, String url) {

L.i("onPageFinished");

closeProgress();

}

public void onReceivedError(WebView view, int errorCode,

String description, String failingUrl) {

L.i("onReceivedError");

closeProgress();

}

}

// 如果不做任何处理,浏览网页,点击系统“Back”键,整个Browser会调用finish()而结束自身,

// 如果希望浏览的网 页回退而不是推出浏览器,需要在当前Activity中处理并消费掉该Back事件。

public boolean onKeyDown(int keyCode, KeyEvent event) {

// if((keyCode==KeyEvent.KEYCODE_BACK)&&webview.canGoBack()){

// webview.goBack();

// return true;

// }

return false;

}

第二步,起线程开始下载文件。

//内部类

private class MyWebViewDownLoadListener implements DownloadListener {

@Override

public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype,

long contentLength) {

if(!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){

Toast t=Toast.makeText(mContext, "需要SD卡。", Toast.LENGTH_LONG);

t.setGravity(Gravity.CENTER, 0, 0);

t.show();

return;

}

DownloaderTask task=new DownloaderTask();

task.execute(url);

}

}

//内部类

private class DownloaderTask extends AsyncTask<String, Void, String> {

public DownloaderTask() {

}

@Override

protected String doInBackground(String... params) {

// TODO Auto-generated method stub

String url=params[0];

//          Log.i("tag", "url="+url);

String fileName=url.substring(url.lastIndexOf("/")+1);

fileName=URLDecoder.decode(fileName);

Log.i("tag", "fileName="+fileName);

File directory=Environment.getExternalStorageDirectory();

File file=new File(directory,fileName);

if(file.exists()){

Log.i("tag", "The file has already exists.");

return fileName;

}

try {

HttpClient client = new DefaultHttpClient();

//                client.getParams().setIntParameter("http.socket.timeout",3000);//设置超时

HttpGet get = new HttpGet(url);

HttpResponse response = client.execute(get);

if(HttpStatus.SC_OK==response.getStatusLine().getStatusCode()){

HttpEntity entity = response.getEntity();

InputStream input = entity.getContent();

writeToSDCard(fileName,input);

input.close();

//                  entity.consumeContent();

return fileName;

}else{

return null;

}

catch (Exception e) {

e.printStackTrace();

return null;

}

}

@Override

protected void onCancelled() {

// TODO Auto-generated method stub

super.onCancelled();

}

@Override

protected void onPostExecute(String result) {

// TODO Auto-generated method stub

super.onPostExecute(result);

closeProgressDialog();

if(result==null){

Toast t=Toast.makeText(mContext, "连接错误!请稍后再试!", Toast.LENGTH_LONG);

t.setGravity(Gravity.CENTER, 0, 0);

t.show();

return;

}

Toast t=Toast.makeText(mContext, "已保存到SD卡。", Toast.LENGTH_LONG);

t.setGravity(Gravity.CENTER, 0, 0);

t.show();

File directory=Environment.getExternalStorageDirectory();

File file=new File(directory,result);

Log.i("tag", "Path="+file.getAbsolutePath());

Intent intent = getFileIntent(file);

startActivity(intent);

}

@Override

protected void onPreExecute() {

// TODO Auto-generated method stub

super.onPreExecute();

showProgressDialog();

}

@Override

protected void onProgressUpdate(Void... values) {

// TODO Auto-generated method stub

super.onProgressUpdate(values);

}

}

[java]view plain copy

//内部类

private class MyWebViewDownLoadListener implements DownloadListener {

@Override

public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype,

long contentLength) {

if(!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){

Toast t=Toast.makeText(mContext, "需要SD卡。", Toast.LENGTH_LONG);

t.setGravity(Gravity.CENTER, 0, 0);

t.show();

return;

}

DownloaderTask task=new DownloaderTask();

task.execute(url);

}

}

//内部类

private class DownloaderTask extends AsyncTask<String, Void, String> {

public DownloaderTask() {

}

@Override

protected String doInBackground(String... params) {

// TODO Auto-generated method stub

String url=params[0];

//          Log.i("tag", "url="+url);

String fileName=url.substring(url.lastIndexOf("/")+1);

fileName=URLDecoder.decode(fileName);

Log.i("tag", "fileName="+fileName);

File directory=Environment.getExternalStorageDirectory();

File file=new File(directory,fileName);

if(file.exists()){

Log.i("tag", "The file has already exists.");

return fileName;

}

try {

HttpClient client = new DefaultHttpClient();

//                client.getParams().setIntParameter("http.socket.timeout",3000);//设置超时

HttpGet get = new HttpGet(url);

HttpResponse response = client.execute(get);

if(HttpStatus.SC_OK==response.getStatusLine().getStatusCode()){

HttpEntity entity = response.getEntity();

InputStream input = entity.getContent();

writeToSDCard(fileName,input);

input.close();

//                  entity.consumeContent();

return fileName;

}else{

return null;

}

} catch (Exception e) {

e.printStackTrace();

return null;

}

}

@Override

protected void onCancelled() {

// TODO Auto-generated method stub

super.onCancelled();

}

@Override

protected void onPostExecute(String result) {

// TODO Auto-generated method stub

super.onPostExecute(result);

closeProgressDialog();

if(result==null){

Toast t=Toast.makeText(mContext, "连接错误!请稍后再试!", Toast.LENGTH_LONG);

t.setGravity(Gravity.CENTER, 0, 0);

t.show();

return;

}

Toast t=Toast.makeText(mContext, "已保存到SD卡。", Toast.LENGTH_LONG);

t.setGravity(Gravity.CENTER, 0, 0);

t.show();

File directory=Environment.getExternalStorageDirectory();

File file=new File(directory,result);

Log.i("tag", "Path="+file.getAbsolutePath());

Intent intent = getFileIntent(file);

startActivity(intent);

}

@Override

protected void onPreExecute() {

// TODO Auto-generated method stub

super.onPreExecute();

showProgressDialog();

}

@Override

protected void onProgressUpdate(Void... values) {

// TODO Auto-generated method stub

super.onProgressUpdate(values);

}

}

第三步,实现一些工具方法。

Java代码 

private ProgressDialog mDialog;

private void showProgressDialog(){

if(mDialog==null){

mDialog = new ProgressDialog(mContext);

mDialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);//设置风格为圆形进度条

mDialog.setMessage("正在加载 ,请等待...");

mDialog.setIndeterminate(false);//设置进度条是否为不明确

mDialog.setCancelable(true);//设置进度条是否可以按退回键取消

mDialog.setCanceledOnTouchOutside(false);

mDialog.setOnDismissListener(new OnDismissListener() {

@Override

public void onDismiss(DialogInterface dialog) {

// TODO Auto-generated method stub

mDialog=null;

}

});

mDialog.show();

}

}

private void closeProgressDialog(){

if(mDialog!=null){

mDialog.dismiss();

mDialog=null;

}

}

public Intent getFileIntent(File file){

//       Uri uri = Uri.parse("http://m.ql18.com.cn/hpf10/1.pdf");

Uri uri = Uri.fromFile(file);

String type = getMIMEType(file);

Log.i("tag", "type="+type);

Intent intent = new Intent("android.intent.action.VIEW");

intent.addCategory("android.intent.category.DEFAULT");

intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

intent.setDataAndType(uri, type);

return intent;

}

public void writeToSDCard(String fileName,InputStream input){

if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){

File directory=Environment.getExternalStorageDirectory();

File file=new File(directory,fileName);

//          if(file.exists()){

//              Log.i("tag", "The file has already exists.");

//              return;

//          }

try {

FileOutputStream fos = new FileOutputStream(file);

byte[] b = new byte[2048];

int j = 0;

while ((j = input.read(b)) != -1) {

fos.write(b, 0, j);

}

fos.flush();

fos.close();

catch (FileNotFoundException e) {

// TODO Auto-generated catch block

e.printStackTrace();

catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}else{

Log.i("tag", "NO SDCard.");

}

}

private String getMIMEType(File f){

String type="";

String fName=f.getName();

/* 取得扩展名 */

String end=fName.substring(fName.lastIndexOf(".")+1,fName.length()).toLowerCase();

/* 依扩展名的类型决定MimeType */

if(end.equals("pdf")){

type = "application/pdf";//

}

else if(end.equals("m4a")||end.equals("mp3")||end.equals("mid")||

end.equals("xmf")||end.equals("ogg")||end.equals("wav")){

type = "audio/*";

}

else if(end.equals("3gp")||end.equals("mp4")){

type = "video/*";

}

else if(end.equals("jpg")||end.equals("gif")||end.equals("png")||

end.equals("jpeg")||end.equals("bmp")){

type = "image/*";

}

else if(end.equals("apk")){

/* android.permission.INSTALL_PACKAGES */

type = "application/vnd.android.package-archive";

}

//      else if(end.equals("pptx")||end.equals("ppt")){

//        type = "application/vnd.ms-powerpoint";

//      }else if(end.equals("docx")||end.equals("doc")){

//        type = "application/vnd.ms-word";

//      }else if(end.equals("xlsx")||end.equals("xls")){

//        type = "application/vnd.ms-excel";

//      }

else{

//        /*如果无法直接打开,就跳出软件列表给用户选择 */

type="*/*";

}

return type;

}

时间: 2024-10-13 02:43:30

WebView中实现文件下载功能的相关文章

asp.net中实现文件下载功能

//TransmitFile实现下载 protected void Button1_Click(object sender, EventArgs e) { /* 微软为Response对象提供了一个新的方法TransmitFile来解决使用Response.BinaryWrite 下载超过400mb的文件时导致Aspnet_wp.exe进程回收而无法成功下载的问题. 代码如下: */ Response.ContentType = "application/x-zip-compressed&quo

springmvc中使用文件下载功能

项目代码:https://github.com/PeiranZhang/springmvc-fileupload 使用文件下载步骤 对请求处理方法使用void或null作为返回类型,并在方法中添加HttpServletResponse参数 将响应的内容类型设为文件的内容类型 添加一个名为Content-Disposition的HTTP响应标题,并赋值attachment; filename= fileName,这里的fileName是默认文件名,应该出现在File Download(文件下载)对

WebView实现文件下载功能

WebView控制调用相应的WEB页面进行展示.安卓源码当碰到页面有下载链接的时候,点击上去是一点反应都没有的.原来是因为WebView默认没有开启文件下载的功能,如果要实现文件下载的功能,需要设置WebView的DownloadListener,通过实现自己的DownloadListener来实现文件的下载.具体操作如下: 1.设置WebView的DownloadListener:     webView.setDownloadListener(new MyWebViewDownLoadLis

Android:WebView中对图片注册上下文菜单

前言 今天一朋友问我一个问题,就是如何在WebView控件中的图片增加上下文菜单,以便增加保存图片等功能.今天就给他简单做了一个演示Demo,现写下来,给有相同问题的朋友提供些许思路吧. 概要实现 其实这个功能很简单,没有太复杂的东西,就是对WebView的控件的使用,一是给WebView注册了上下文菜单事件,二是在响应事件中去判断事件源的类型,如果是图片类型,则把url取出来 注册上下文菜单事件 这个就比较简单了通过下面的代码即可完成. WebView vw = (WebView) findV

php实现文件下载功能总结

php实现文件下载功能总结 其实所谓的下载按钮根本就是个A标签,里面记录着要下载文件的路径,如果要下载的文件恰巧是图片或者是文本文件,那么下载的时候就会被浏览器弹出一个新窗口或者直接页面跳转掉并且直接打开这个文件,解决方法是将下载的文件改成压缩后的格式或者是浏览器不能直接打开的格式. 下面我来说一下下载的流程:浏览器发送一个请求,请求访问服务器中的某个网页(如:down.php),服务器接受到该请求以后,马上运行该down.php文件,运行该文件的时候,必然要把将要被下载的文件读入内存当中,这里

Android安全开发之WebView中的地雷

0X01 About WebView 在Android开发中,经常会使用WebView来实现WEB页面的展示,在Activiry中启动自己的浏览器,或者简单的展示一些在线内容等.WebView功能强大,应用广泛,但它是天使与恶魔的合体,一方面它增强了APP的上网体验,让APP功能更多样化,另一方面它也引入了很多的安全问题.在过去几年WebView中被披露的重大漏洞包括了任意代码执行漏洞.跨域.密码明文保存等,这些安全问题可以直接导致用户敏感信息泄露,移动终端被恶意攻击者控制.下文将详细介绍这一系

JAVA文件下载功能问题解决日志

今天给报告系统做了个下载功能,遇到了挺多问题,通过查资料一一解决了. 1.首先遇到的问题是:java后台的输出流输出之后,没有任何报错,浏览器端不弹出保存文件的对话框,原本是ajax请求到后台的controller方法中添加了下载的方法,type和async两个参数的四种组合都不行,弃用ajax,用window.location.href='file/download?path='+file;重新发一个新的下载请求之后,保存对话框终于弹出. 2.弹出之后,发现文件名乱码,后台的解决方案代码如下:

java web文件下载功能实现

需求:实现一个具有文件下载功能的网页,主要下载压缩包和图片 两种实现方法: 一:通过超链接实现下载 在HTML网页中,通过超链接链接到要下载的文件的地址 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Insert title here</title> </head> <body> <h1>通过链接下载文件&

java—servlet实现文件下载功能

最近在学javaweb技术,利用所学知识做了一个简单的项目,在服务器上实现给用户下载文件的功能.自动列出文件夹下的文件提供下载,支持中文文件名. 结果图 其中使用了tomcat来部署服务器,代码中应用了少许EL表达式和JSTL标签,使用了jsp和servlet,当然还有java和http的基础知识,但是个人觉得重点是http的Content-Disposition头信息和Content-type头信息在servlet中下载方面的应用,详见DownloadPageServlet.java源码. 关