【发布时间】:2016-09-07 12:29:43
【问题描述】:
我的问题的症结在于我需要异步使用 datatransferitemlist,这与规范中描述的功能不一致,即一旦事件结束,您将被锁定在 dataTransfer.items 集合之外。
https://bugs.chromium.org/p/chromium/issues/detail?id=137231 http://www.whatwg.org/specs/web-apps/current-work/multipage/dnd.html#drag-data-store
案犯如下。在下面对我的问题和想法进行更详细的描述。
drophandler: function(event) {
event.stopPropagation();
event.preventDefault();
event.dataTransfer.dropEffect = 'copy';
zip.workerScriptsPath = "../bower_components/zip.js/WebContent/";
zip.useWebWorkers = false; // Disabled because it just makes life more complicated
// Check if files contains just a zip
if (event.dataTransfer.files[0].name.match(/(?:\.([^.]+))?$/) == 'zip') {
var reader = new FileReader();
that = this;
reader.onload = function(e) {
that.fire('zipuploaded', e.target.result.split(',')[1]);
}
reader.readAsDataURL(event.dataTransfer.files[0]);
// Rev up that in browser zipping
} else {
var that = this;
var items = event.dataTransfer.items;
// Async operation, execution falls through from here
zip.createWriter(new zip.Data64URIWriter(), function(writer) {
(function traverse(list, path, i, depth) {
return new Promise(function(resolve, reject) {
var item;
if (depth == 0) {
if (i == list.length) {
writer.close(function(uri) {
that.fire('zipuploaded', uri.split(',')[1]); // Just the base64, please
fulfill(1);
return;
});
} else {
console.log(i);
console.log(list);
var item = list[i].webkitGetAsEntry();
}
} else {
if (i == list.length) {
resolve(0);
return;
} else {
item = list[i];
}
}
if (item.isFile) {
item.file(function(file) {
// Zipping operations done asynchronously, it'll fail by roughly the second operation
writer.add(path + file.name, zip.BlobReader(file), function() {
traverse(list, path, i + 1, depth).then(resolve(0)); // Next item
});
});
} else if (item.isDirectory) {
var dirReader = item.createDirReader();
dirReader.readEntries(function(entries) {
// Operate on child folder then the next item at this level
traverse(entries, path + item.name + "/", 0, depth + 1).then(function() {
traverse(list, path, i + 1, depth).then(resolve(0));
});
});
}
});
})(items, "", 0, 0); // Begin with datatransferitemlist, 0th element, depth 0
});
this.$.uploadarea.classList.remove('highlightdrag');
}
// When we exit it kills the event.dataTransfer.items
},
我正在使用与 HTML5 DnD API 异步的 zip.js。 ondrop 事件在异步 zip.createWriter/writer.add 操作完成之前结束。我可以想出四种方法来解决这个问题,虽然我不知道如何实现它们并希望得到一些建议。
- 阻塞直到 createWriter 完成。 (阻止 javascript?呃)
- 防止 ondrop 将我锁定在 dataTransfer.items 之外(这似乎是为了安全所以不太可能)
- 先同步复制出dataTransfer.items的内容(可能很慢)
- 同步执行所有操作(不要认为 zip.js 允许这样做,JsZip 可以,但我放弃了这一点,因为它对大型文件集有其自身的限制)
【问题讨论】:
-
一个问题是您在
traverse(list, path, i + 1, depth).then(resolve(0))立即调用resolve函数,而不是引用resolve函数。不确定预期的结果是什么?将所有删除的文件添加到单个.zip文件夹?
标签: javascript html asynchronous drag-and-drop zip.js