我正在使用Vuejs和DataTransfer异步上传文件,并且我希望允许拖放多个文件以一次上传。

我可以进行第一次上载,但是在完成上载时,Javascript已垃圾回收或更改了DataTransfer items对象。

我该如何重做(或克隆event/DataTransfer对象),以便在整个ajax调用中仍然可以使用数据?

我已经关注了MDN文档中有关如何使用DataTransfer的信息,但是我很难将其应用于我的特定情况。正如您在我的代码中看到的那样,我还尝试了复制事件对象,但是显然,它没有进行深层复制,只是传递了引用,这没有帮助。

    methods: {
        dropHandler: function (event) {
            if (event.dataTransfer.items) {
                let i = 0;
                let self = this;
                let ev = event;

                function uploadHandler() {
                    let items = ev.dataTransfer.items;
                    let len = items.length;

                    // len NOW EQUALS 4

                    console.log("LEN: ", len);
                    if (items[i].kind === 'file') {
                        var file = items[i].getAsFile();
                        $('#id_file_name').val(file.name);
                        var file_form = $('#fileform2').get(0);
                        var form_data = new FormData(file_form);

                        if (form_data) {
                            form_data.append('file', file);
                            form_data.append('type', self.type);
                        }

                        $('#file_progress_' + self.type).show();
                        var post_url = '/blah/blah/add/' + self.object_id + '/';
                        $.ajax({
                            url: post_url,
                            type: 'POST',
                            data: form_data,
                            contentType: false,
                            processData: false,
                            xhr: function () {
                                var xhr = $.ajaxSettings.xhr();
                                if (xhr.upload) {
                                    xhr.upload.addEventListener('progress', function (event) {
                                        var percent = 0;
                                        var position = event.loaded || event.position;
                                        var total = event.total;
                                        if (event.lengthComputable) {
                                            percent = Math.ceil(position / total * 100);
                                            $('#file_progress_' + self.type).val(percent);
                                        }
                                    }, true);
                                }
                                return xhr;
                            }
                        }).done((response) => {
                                i++;
                                if (i < len) {

                                    // BY NOW, LEN = 0.  ????

                                    uploadHandler();
                                } else {
                                    self.populate_file_lists();
                                }
                            }
                        );
                    }
                }

                uploadHandler();
            }
        },

最佳答案

调用await后,您将不再位于该函数的原始调用堆栈中。这在事件监听器中尤其重要。

我们可以使用setTimeout重现相同的效果:

dropZone.addEventListener('drop', async (e) => {
  e.preventDefault();
  console.log(e.dataTransfer.items);
  setTimeout(()=> {
    console.log(e.dataTransfer.items);
  })
});

例如,拖动四个文件将输出:
DataTransferItemList {0: DataTransferItem, 1: DataTransferItem, 2: DataTransferItem, 3: DataTransferItem, length: 4}
DataTransferItemList {length: 0}

事件发生后,状态已更改,并且项目已丢失

有两种方法可以解决此问题:
  • 复制项目并对其进行迭代
  • 将异步作业(Promises)推送到数组中,并稍后使用Promise.all对其进行处理

  • 第二种解决方案比在循环中使用await更直观。另外,考虑parallel connections are limited。使用数组,您可以创建块以限制同时上传。

    function pointlessDelay() {
      return new Promise((resolve, reject) => {
        setTimeout(resolve, 1000);
      });
    }
    
    const dropZone = document.querySelector('.dropZone');
    
    dropZone.addEventListener('dragover', (e) => {
      e.preventDefault();
    });
    
    dropZone.addEventListener('drop', async (e) => {
      e.preventDefault();
      console.log(e.dataTransfer.items);
      const queue = [];
    
      for (const item of e.dataTransfer.items) {
        console.log('next loop');
        const entry = item.webkitGetAsEntry();
        console.log({item, entry});
        queue.push(pointlessDelay().then(x=> console.log(`${entry.name} uploaded`)));
      }
    
      await Promise.all(queue);
    });
    body {
      font-family: sans-serif;
    }
    
    .dropZone {
      display: inline-flex;
      background: #3498db;
      color: #ecf0f1;
      border: 0.3em dashed #ecf0f1;
      border-radius: 0.3em;
      padding: 5em;
      font-size: 1.2em;
    }
    <div class="dropZone">
      Drop Zone
    </div>

    10-06 07:54