Суть моей проблемы в том, что мне нужно использовать асинхронно 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
},
Я использую zip.js, который является асинхронным с API-интерфейсом HTML5 DnD. Событие ondrop заканчивается до завершения операций асинхронного zip.createWriter/writer.add. Я могу подумать о четырех способах решения этой проблемы, хотя я не знаю, как реализовать любой из них, и хотел бы получить некоторые советы.
- Заблокировать до создания createWriter. (Блокировка javascript? Uhoh)
- Предотвратите блокировку ядро из dataTransfer.items(кажется, для безопасности так маловероятно)
- Синхронно копировать содержимое dataTransfer.items сначала (возможно, очень медленно)
- Делать все синхронно (не думаю, что zip.js позволяет это, JsZip делает, но я отошел от этого из-за того, что он имел свои ограничения с большими файловыми наборами)