Android的webview:像浏览器一样下载文件

我正在使用webview指向另一个团队的dynamic网站的Android应用程序。
当我下载一个文件(大部分是dynamicredirect的 PDF和ZIP), 我得到的是一个文件在下载文件夹中包含一些消息,如“用户不允许读取文件”的HTML代码 ,不pipe我如何实现下载,我尝试:

  1. 下载pipe理器
  2. 意图(让外部浏览器来pipe理下载)
  3. “手工”(AsyncTask和httpconnection …)

所有的结果都一样。

使用普通的浏览器进行浏览可以在台式电脑,安卓和iOS设备上正常工作

为什么WebView不应该有权访问文件?

可能是会话问题? http端口?
我真的需要一些想法…

另一个提示:当从同一链接下载两个文件时,链接将redirect到相同的文件,但导致两个不同的文件名…

编辑 :而不是指向的WebView的Web应用程序,我试图指向一个链接redirect下载另一个文件的公共网页,以及,只是它的工作原理。

这里是webview.setDownloadListener - onDownloadStart()参数:

  userAgent=Mozilla/5.0 (Linux; Android 4.4.2; Nexus 7 Build/KOT49H) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/30.0.0.0 Safari/537.36 contentDisposition=attachment; filename=correct_filename.pdf, url=http://www.xxx.xx/site/downloadfile.wplus?REDIRECTFILE=D-507497120&ID_COUNTOBJ=ce_5_home&TYPEOBJ=CExFILE&LN=2 mimeType=application/octet-stream 

这是一些代码

  wv.getSettings().setSupportMultipleWindows(true); wv.getSettings().setJavaScriptCanOpenWindowsAutomatically(true); wv.getSettings().setAllowFileAccess(true); wv.getSettings().setJavaScriptEnabled(true); wv.getSettings().setBuiltInZoomControls(true); wv.getSettings().setDisplayZoomControls(false); wv.getSettings().setLoadWithOverviewMode(true); wv.getSettings().setUseWideViewPort(true); wv.setDownloadListener(new DownloadListener() { @Override public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype, long contentLength){ DownloadManager.Request request = new DownloadManager.Request(Uri.parse(url)); request.setDescription("Download file..."); request.setTitle(URLUtil.guessFileName(url, contentDisposition, mimetype)); request.allowScanningByMediaScanner(); request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); //Notify client once download is completed! request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, URLUtil.guessFileName(url, contentDisposition, mimetype)); DownloadManager dm = (DownloadManager) getSystemService(DOWNLOAD_SERVICE); dm.enqueue(request); Toast.makeText(getApplicationContext(), "Downloading File", Toast.LENGTH_LONG).show(); } } 

编辑II

以下是我在“手工”下载文件时使用的代码:

onDownloadStart()是我在哪里调用downloadFileAsync():

  public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimeType, long contentLength) { String fileName; try { fileName = URLUtil.guessFileName(url, contentDisposition, mimeType); downloadFileAsync(url, fileName); }catch (Exception e){ } } 

这是AsyncTask:

 private void downloadFileAsync(String url, String filename){ new AsyncTask<String, Void, String>() { String SDCard; @Override protected void onPreExecute() { super.onPreExecute(); } @Override protected String doInBackground(String... params) { try { URL url = new URL(params[0]); HttpURLConnection urlConnection = null; urlConnection = (HttpURLConnection) url.openConnection(); urlConnection.setRequestMethod("GET"); urlConnection.setDoOutput(true); urlConnection.connect(); int lengthOfFile = urlConnection.getContentLength(); //SDCard = Environment.getExternalStorageDirectory() + File.separator + "downloads"; SDCard = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS)+""; int k = 0; boolean file_exists; String finalValue = params[1]; do { if (k > 0) { if (params[1].length() > 0) { String s = params[1].substring(0, params[1].lastIndexOf(".")); String extension = params[1].replace(s, ""); finalValue = s + "(" + k + ")" + extension; } else { String fileName = params[0].substring(params[0].lastIndexOf('/') + 1); String s = fileName.substring(0, fileName.lastIndexOf(".")); String extension = fileName.replace(s, ""); finalValue = s + "(" + k + ")" + extension; } } File fileIn = new File(SDCard, finalValue); file_exists = fileIn.exists(); k++; } while (file_exists); File file = new File(SDCard, finalValue); FileOutputStream fileOutput = null; fileOutput = new FileOutputStream(file, true); InputStream inputStream = null; inputStream = urlConnection.getInputStream(); byte[] buffer = new byte[1024]; int count; long total = 0; while ((count = inputStream.read(buffer)) != -1) { total += count; //publishProgress(""+(int)((total*100)/lengthOfFile)); fileOutput.write(buffer, 0, count); } fileOutput.flush(); fileOutput.close(); inputStream.close(); } catch (MalformedURLException e){ } catch (ProtocolException e){ } catch (FileNotFoundException e){ } catch (IOException e){ } catch (Exception e){ } return params[1]; } @Override protected void onPostExecute(final String result) { } }.execute(url, filename); } 

摘自如何从webview中的dynamicurl下载PDF
感谢名单

Solutions Collecting From Web of "Android的webview:像浏览器一样下载文件"

最后,我决定从Android股票浏览器代码寻找DownloadHandler 。 我的代码中唯一明显的缺点是cookie (!!!)。

这是我的最终工作版本(DownloadManager方法):

  wv.setDownloadListener(new DownloadListener() { @Override public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimeType, long contentLength) { DownloadManager.Request request = new DownloadManager.Request(Uri.parse(url)); request.setMimeType(mimeType); //------------------------COOKIE!!------------------------ String cookies = CookieManager.getInstance().getCookie(url); request.addRequestHeader("cookie", cookies); //------------------------COOKIE!!------------------------ request.addRequestHeader("User-Agent", userAgent); request.setDescription("Downloading file..."); request.setTitle(URLUtil.guessFileName(url, contentDisposition, mimeType)); request.allowScanningByMediaScanner(); request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, URLUtil.guessFileName(url, contentDisposition, mimeType)); DownloadManager dm = (DownloadManager) getSystemService(DOWNLOAD_SERVICE); dm.enqueue(request); Toast.makeText(getApplicationContext(), "Downloading File", Toast.LENGTH_LONG).show(); } }); 
 wv.setDownloadListener(new DownloadListener() { @Override public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimeType, long contentLength) { DownloadManager.Request request = new DownloadManager.Request(Uri.parse(url)); request.setMimeType(mimeType); //------------------------COOKIE!!------------------------ String cookies = CookieManager.getInstance().getCookie(url); request.addRequestHeader("cookie", cookies); //------------------------COOKIE!!------------------------ request.addRequestHeader("User-Agent", userAgent); request.setDescription("Downloading file..."); request.setTitle(URLUtil.guessFileName(url, contentDisposition, mimeType)); request.allowScanningByMediaScanner(); request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, URLUtil.guessFileName(url, contentDisposition, mimeType)); DownloadManager dm = (DownloadManager) getSystemService(DOWNLOAD_SERVICE); dm.enqueue(request); Toast.makeText(getApplicationContext(), "Downloading File", Toast.LENGTH_LONG).show(); } }); 

感谢jc为你错过的答案); 在代码结束..

AsyncTask手动检索文件名(您可以添加额外的代码到这个类):

 private static class getFileNameAsync extends AsyncTask<String, Void, String> { @Override protected void onPreExecute() { super.onPreExecute(); } @Override protected String doInBackground(String... params) { URL url; String filename = null; HttpURLConnection conn = null; try { url = new URL(params[0]); conn = (HttpURLConnection) url.openConnection(); conn.connect(); conn.setInstanceFollowRedirects(false); try { for(int i = 0; i < 100; i++) { String stringURL = conn.getHeaderField("Location"); if (stringURL != null) { url = new URL(stringURL); conn = (HttpURLConnection) url.openConnection(); conn.connect(); conn.setInstanceFollowRedirects(false); } else { i = 100; } } } catch (Exception e) { e.printStackTrace(); } String depo = conn.getHeaderField("Content-Disposition"); if (depo != null) { String depoSplit[] = depo.split(";"); int size = depoSplit.length; for(int i = 0; i < size; i++) { if(depoSplit[i].startsWith("filename=")) { filename = depoSplit[i].replaceFirst("(?i)^.*filename=\"?([^\"]+)\"?.*$", "$1").trim(); i = size; } } } } catch (MalformedURLException e){ e.printStackTrace(); } catch (ProtocolException e){ e.printStackTrace(); } catch (FileNotFoundException e){ e.printStackTrace(); } catch (IOException e){ e.printStackTrace(); } catch (Exception e){ e.printStackTrace(); } finally { if (conn != null) conn.disconnect(); } return filename; } @Override protected void onPostExecute(String filename) { super.onPostExecute(filename); } } 

DownloadListener(它是稳定的版本,不要改变它):

  mWebView.setDownloadListener(new DownloadListener() { @Override public void onDownloadStart(String url, String userAgent, String contentDisposition, String mimetype, long contentLength) { if(Build.VERSION.SDK_INT >=23){ Context nContext = MainActivity.this.getApplicationContext(); if(nContext.checkSelfPermission(android.Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){ MainActivity.this.requestPermissions(new String[]{android.Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1); return; } } String fileName = ""; url = url.replace(" ", "%20"); AsyncTask<String, Void, String> asyncTask = new getFileNameAsync(); asyncTask.execute(url); try { fileName = asyncTask.get(); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } catch (CancellationException e) { e.printStackTrace(); } if ((fileName == null) || (fileName.hashCode() == "".hashCode())) { fileName = URLUtil.guessFileName(url, contentDisposition, mimetype); } DownloadManager.Request request = new DownloadManager.Request(Uri.parse(url)); request.setMimeType(mimetype); String cookies = CookieManager.getInstance().getCookie(url); request.addRequestHeader("Cookie", cookies); request.addRequestHeader("User-Agent", userAgent); request.setDescription("Downloading File"); request.setTitle(fileName); request.allowScanningByMediaScanner(); request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, fileName); DownloadManager downloadManager = (DownloadManager) getSystemService(DOWNLOAD_SERVICE); if (downloadManager != null) { downloadManager.enqueue(request); } Toast.makeText(getApplicationContext(), "Downloading File", Toast.LENGTH_LONG).show(); } });