В веб-представлении Android Lollipop я хочу, чтобы пользователь загрузил сгенерированный файл txt:
// Store some text in a data URL.
var dataUrl = (window.URL || window.webkitURL).createObjectURL(
new Blob(["Hello world. :)"]));
// Create a link that lets the user download the text file as hello.txt.
var downloadLink = document.createElement('a');
downloadLink.setAttribute('href', dataUrl);
downloadLink.innerHTML = 'Click to download.';
// Working with David on this. I did the same thing.
// Fyi, setting the 'download' attribute just makes hitting the link
// nop, ie. do nothing at all in the web view, so I omitted it.
// - John
// Display the link.
document.getElementById('container').appendChild(downloadLink);
На стороне Java java я написал DownloadListener, который пытается загрузить DownloadManager для загрузки файла:
package com.somesideprojects;
import android.webkit.DownloadListener;
/**
* A download listener that lets users download files from the web view.
*/
public class CustomDownloadListener implements DownloadListener {
private MainActivity currentActivity;
@Override
public void onDownloadStart(
String url,
String userAgent,
String contentDisposition,
String mimeType,
long contentLength) {
android.util.Log.d("Logger",
"url : " + url +
" userAgent: " + userAgent +
" contentDisposition: " + contentDisposition +
" mimeType: " + mimeType + " contentLength " + contentLength);
android.net.Uri source = android.net.Uri.parse(url);
// Make a new request.
android.app.DownloadManager.Request request =
new android.app.DownloadManager.Request(source);
// Appears the same in notification bar while downloading.
String filename = getFilename(contentDisposition);
request.setDescription(
"This project will be saved in your downloads folder as " + filename + ".");
request.setTitle(filename);
// Add cookie on request header (for authenticated web app).
String cookieContent = getCookieFromAppCookieManager(source.getHost());
request.addRequestHeader("Cookie", cookieContent);
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.HONEYCOMB) {
request.allowScanningByMediaScanner();
request.setNotificationVisibility(
android.app.DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
}
// Save the file in the "Downloads" folder of SDCARD.
request.setDestinationInExternalPublicDir(
android.os.Environment.DIRECTORY_DOWNLOADS, filename);
// Get the download service and enqueue the file.
android.app.DownloadManager manager =
(android.app.DownloadManager) this.currentActivity.getApplication()
.getSystemService(android.content.Context.DOWNLOAD_SERVICE);
manager.enqueue(request);
}
public String getFilename(String contentDisposition){
String filename[] = contentDisposition.split("filename=");
return filename[1].replace("filename=", "").replace("\"", "").trim();
};
public String getCookieFromAppCookieManager(String url){
android.webkit.CookieManager cookieManager = android.webkit.CookieManager.getInstance();
if (cookieManager == null) {
return null;
}
String rawCookieHeader = null;
// Extract Set-Cookie header value from Android app CookieManager for this URL
rawCookieHeader = cookieManager.getCookie(url);
if (rawCookieHeader == null) {
return null;
}
return rawCookieHeader;
};
public void setCurrentActivity(MainActivity currentActivity) {
this.currentActivity = currentActivity;
}
}
Эта ошибка java возникает, когда я нажимаю ссылку в веб-представлении:
java.lang.IllegalArgumentException: Can only download HTTP/HTTPS URIs: blob:file%3A///f566c1cf-b0b2-4382-ba16-90bab359fcc5
at android.app.DownloadManager$Request.<init>(DownloadManager.java:429)
Я получил ту же ошибку, когда размещал свою страницу на HTTP-сервере, поэтому ошибка, похоже, связана с частью blob:
.
Каковы мои варианты сейчас? Как загрузить данные, хранящиеся на URL-адресе? В конечном счете, я хочу загрузить намного больший blob (~ 50 МБ).
Я могу передать данные объекту, введенному в addJavascriptInterface, но мне нужно base64 закодировать мой blob и декодировать на стороне java (поскольку могут быть переданы только примитивы и простые типы). Кодирование в base64 с помощью javascript сбой Chromium для 50 МБ blobs (я получаю ошибки из памяти при вызове readAsDataURL).
Как еще я могу загрузить 50-мегабайтный кадр из веб-представления? Или передать двоичные данные 50 МБ из javascript в Android java?
== Подробная информация о случае использования ==
Я использую javascript для генерации 50-мегабайтного видео файла с типом mime video/x-ms-wmv
. Это поколение полностью клиентское. Я уверен, что этот шаг работает, так как я могу загрузить файл в браузере рабочего стола (а также через обычное приложение Chrome) с помощью URL-адреса объекта. Затем я хочу, чтобы пользователь мог сохранить этот файл на своем внешнем хранилище (возможно, в DCIM или Downloads).
В несвязанной заметке я хотел бы также сделать то же самое с аудио файлами WAV (например, для создания мелодий звонка для сохранения в мелодии звонка).