拖放文件上传无需AJAX,在前台同步

Drag-and-drop file uploading without AJAX, synchronously in the foreground?

本文关键字:前台 同步 AJAX 文件 拖放      更新时间:2023-09-26

我有一个网站,定期上传<input type="file">文件,在提交表单时将数据张贴到后端。

我想逐步增强表单,这样你就可以从浏览器外的视口中的任何地方(而不仅仅是文件输入字段上,就像一些浏览器中内置的那样)放置文件来上传

表单是否自动提交并不重要。因此,如果拖放只选择文件字段中的文件,而不开始上传,那也没关系。我不需要对多个文件的支持。我不需要显示上传进度,缩略图或任何花哨的东西。

我知道有些JS库支持拖放上传,但它们似乎都是通过AJAX上传的。我可以这样做,但我需要修改后端和前端来处理上传错误、重定向和显示成功的正确消息等等

我想要一个不需要任何后端更改的渐进式增强。它应该使用页面中的表单同步进行。JS是好的,只要上传发生在"前台"。当然,同步AJAX不起作用。

虽然不是真正的"同步"(JavaScript执行实际上不会停止),但您可以用程序设置<input type="file">选择的文件。事实上,这些元素和拖动共享它们的文件后端实现(FileFileList实例),所以它实际上是直接的。更重要的是,由于两个前端都使用FileList,拖动多个文件也同样无缝。

这适用于Chrome(使用jQuery):http://jsfiddle.net/qMmPr/.

$(document).on("dragover drop", function(e) {
    e.preventDefault();  // allow dropping and don't navigate to file on drop
}).on("drop", function(e) {
    $("input[type='file']")
        .prop("files", e.originalEvent.dataTransfer.files)  // put files into element
        .closest("form")
          .submit();  // autosubmit as well
});

感谢@pimvdb的评论,我提出了一个非常优雅的解决方案。

既然在<input type="file" />上拖放是可行的,为什么不在dragstart上全屏显示以确保用户不会错过它呢?不管怎样,他一直在拖延,所以此刻他的意图是明确的。

下面是一个演示:https://jsfiddle.net/08wbo4um

注意:不幸的是,这似乎在iframe中不起作用,但它确实在实际页面上起作用。你仍然可以理解这种行为。

以下是片段:

  $('input[type="file"]').on('change', function(e){
    var fileName = e.target.files[0].name;
    if (fileName) {
      $(e.target).parent().attr('data-message', fileName);
    }
  });
  
  $(document).on('drag dragstart dragend dragover dragenter dragleave drop', function(e) {
    if ($('input[type="file"]').length) {
      if (['dragover', 'dragenter'].indexOf(e.type) > -1) {
        if (window.dragTimeout)
          clearTimeout(window.dragTimeout);
        $('body').addClass('dragged');
      } else if (['dragleave', 'drop'].indexOf(e.type) > -1) {
        // Without the timeout, some dragleave events are triggered
        // when the :after appears, making it blink...
        window.dragTimeout = setTimeout(function() {
          $('body').removeClass('dragged');
        }, 100);
      }
    }
  });
h3, p {
  text-align: center;
}
.form-group {
  margin: 30px;
}
.file-upload .form-control {
  height: 150px;
  outline: 1px dashed #ccc;
  outline-offset: -15px;
  background-color: #eee;
}
.file-upload .form-control:before {
  content: "'f093";
  font: normal normal normal 14px/1 FontAwesome;
  font-size: 3em;
  left: 0;
  right: 0;
  display: block;
  margin: 20px auto;
  text-align: center;
}
.file-upload .form-control:after {
  content: attr(data-message);
  left: 0;
  right: 0;
  bottom: 0;
  text-align: center;
  display: block;
}
.file-upload .form-control input[type="file"] {
  cursor: pointer;
  opacity: 0;
  width: 100%;
  height: 100%;
  position: absolute;
  top: 0;
  bottom: 0;
  right: 0;
  left: 0;
}
body.dragged .file-upload .form-control input[type="file"] {
  /* Make sure it is full screen, whatever the position absolute container */
  position: fixed;
  top: -50vh;
  bottom: -50vh;
  left: -50vw;
  right: -50vw;
  height: 200vh;
  width: 200vw;
  z-index: 10002;
}
body:after {
  content: 'You can drop the file. :-)';
  font-size: 2em;
  text-align: center;
  line-height: 100vh;
  position: absolute;
  top: 10px;
  bottom: 10px;
  left: 10px;
  right: 10px;
  background-color: #eee;
  z-index: 10000;
  border-radius: 4px;
  border: thin solid #ccc;
  visibility: hidden;
  opacity: 0;
  transition: visibility 0s, opacity 0.5s ease;
}
body.dragged:after {
  opacity: 1;
  visibility: visible;
}
<link href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet"/>
<link href="https://maxcdn.bootstrapcdn.com/font-awesome/4.7.0/css/font-awesome.min.css" rel="stylesheet"/>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.0/jquery.min.js"></script>
<h3>Drag N Drop file upload without AJAX Demo</h3>
<p>Try drag and dropping a file. :-)</p>
<div class="form-group file-upload" required="required">
    <label class="cols-sm-2 control-label" for="document_file">File Upload</label><br>
    <div class="cols-sm-10">
      <div class="input-group">
        <span class="input-group-addon"><i class="fa fa-file" aria-hidden="true"></i></span>
        <div class="form-control" data-message="Click to select file or drag n drop it here">
          <input required="required" title="Click to select file or drag n drop it here" type="file" name="document[file]" id="document_file">
        </div>
      </div>
    </div>
  </div>

这可以通过将autoUpload设置为false,收集数组中的文件,然后在表单提交时对所有文件和表单数据进行一次ajax调用来完成,如下所述。