我正在尝试在Chrome扩展程序中下载多个文件。以下代码创建指向文件的虚拟链接,然后触发下载文件的.click()事件。 问题是只有第一个.click()事件才会触发下载。后续的.click()事件将被忽略。
这里是 manifest.json :
{
"name": "Simple File Downloader",
"version": "0.1",
"permissions": ["contextMenus", "http://*/"],
"background": {
"persistent": false,
"scripts": ["sample.js"]
},
"content_security_policy": "script-src 'self'; object-src 'self'",
"manifest_version": 2
}
这里是 sample.js :
function onClickHandler(info, tab) {
var a = document.createElement('a');
a.href = 'http://or.cdn.sstatic.net/chat/so.mp3';
a.download = 'so.mp3';
document.body.appendChild(a);
a.click(); // this click triggers the download
// this timeout violates content security policy
// setTimeout(a, 300);
a.click(); // this click doesn't do anything
document.body.removeChild(a);
a = document.createElement('a');
a.href = 'http://or.cdn.sstatic.net/chat/so.mp3';
a.download = 'so.mp3';
document.body.appendChild(a);
a.click(); // this click doesn't do anything either
document.body.removeChild(a);
};
chrome.contextMenus.onClicked.addListener(onClickHandler);
chrome.runtime.onInstalled.addListener(function() {
chrome.contextMenus.create({"title": "Download File", "id":"download_file"});
});
我试过了:
使用FileSaver.js在Chrome Extension write to file system中描述的下载文件的不同方法,结果完全相同,第一个文件被下载,第二个文件没有
按照Is it possible to make two .click method calls in javascript中的说明添加超时,导致内容安全政策违规,我尝试使用Content-Security-Policy error in google chrome extension making中描述的方法进行解决,但没有成功
使用JQuery click event works only once中描述的jQuery .live方法,也没有成功,但我并不是100%肯定我正确实现了这个(如果人们认为这种方法应该解决,可以稍后发布代码它)
很惊讶为什么简单地保存多个文件太难了。感谢任何帮助。
答案 0 :(得分:5)
诀窍不是使用element.click
方法而是创建多个MouseEvent
。为此,您需要在每次需要点击时创建一个MouseEvent
。
function clicker(el, clickCount) {
var mousedownEvent;
while(clickCount--) {
mousedownEvent = document.createEvent("MouseEvent");
mousedownEvent.initMouseEvent("click", true, true, window, 0, null, null, null, null, false , false, false, false, 0, null);
el.dispatchEvent(mousedownEvent);
}
}
clicker(a, 3);
// your anchor 'a' gets clicked on 3 times.
在Chrome中使用此方法时,您会收到来自浏览器的警告,询问“此网站正在尝试下载多个文件。您要允许这样做吗?[拒绝] [允许]” 。因此,如果您在扩展程序的后台页面中执行此操作,则后台页面会收到警告,用户无法看到它,因此用户无法单击“允许”。
(粗略/讨厌)解决方法是创建一个“点击”锚点的选项卡。像这样:
function _anchorDownloader(url, filename) {
var timeout = 500;
return 'javascript:\'<!doctype html><html>'+
'<head></head>' +
'<script>' +
'function initDownload() {'+
'var el = document.getElementById("anchor");'+
'el.click();' +
'setTimeout(function() { window.close(); }, ' + timeout + ');' +
'}'+
'</script>' +
'<body onload="initDownload()">' +
'<a id="anchor" href="' + url + '" download="'+ filename + '"></a>'+
'</body>' +
'</html>\'';
};
function downloadResource(info, tab) {
// ...
chrome.tabs.create( { 'url' : _anchorDownloader( url, filename ), 'active' : false } );
// ...
}
chrome.contextMenus.create({"title": "Save Image…", "contexts":["image"], "onclick": downloadResource });
要实现此功能,扩展程序必须在 manifest.json 中将"tabs"
作为permission
。您可以调整超时以关闭选项卡,但是,如果关闭它太快,则不会发生下载。
答案 1 :(得分:0)
不要使用不再推荐的.live()
方法,而是尝试.on()
$(document).on("click", "a", function( event ){
// do whatever
});