服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Android - Android WebView实现文件下载功能

Android WebView实现文件下载功能

2022-02-24 15:16摄氏三十七度 Android

这篇文章主要为大家详细介绍了Android WebView实现文件下载功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

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

1、设置WebView的DownloadListener: 

?
1
webView.setDownloadListener(new MyWebViewDownLoadListener());

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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进行一系列设置。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
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;
}

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
//内部类
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);
  }
 
}

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
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;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://blog.csdn.net/cc20032706/article/details/69525802

延伸 · 阅读

精彩推荐