在 Google Chrome/Chromium 和 Safari 中拖放文件上传?

Posted

技术标签:

【中文标题】在 Google Chrome/Chromium 和 Safari 中拖放文件上传?【英文标题】:Drag-and-drop file upload in Google Chrome/Chromium and Safari? 【发布时间】:2011-02-09 02:27:18 【问题描述】:

在 Firefox 3.6 中可以进行拖放文件上传。

A Google search for html5 drag-and-drop file uploading -gmail 给出如下内容:

Native Drag + Drop file upload in Firefox 3.6 http://www.appelsiini.net/2009/10/html5-drag-and-drop-multiple-file-upload http://www.thecssninja.com/javascript/drag-and-drop-upload

所有这些指南都使用 FileReader(或 Firefox 3.6 已弃用的 getAsBinary,其他浏览器也不支持)。

不过,Google 最近发布了 Gmail 更新,允许在 Chromium 和 Firefox 和 Chromium does not have FileReader 中拖放文件上传。我用的是最新的Chromium nightly,可以拖拽上传文件,但不支持FileReader

我看到有人提到可以通过拖放到<input type="file" /> 上进行拖放上传,但是一次只能支持一个文件,而 Gmail 的上传器可以处理多个文件被拖放到上面,所以这是显然不是他们在做什么。

所以问题是,他们是如何做到的?您如何支持 Chromium 进行 HTML5 文件上传?另外,能支持Safari吗?

【问题讨论】:

P.S.阅读源代码非常困难,因为 Gmail 的源代码由于他们对其进行的所有优化而被很好地混淆了。 P.P.S. HTML <input type="file" />FileList 是只读的,因此您也不能将拖放的文件移植到其中一个。 这篇文章的标题可以改吗?我刚刚发布了一个重复的问题,询问“Chrome”中的“拖放”,但它与这个问题不匹配,因为标题使用 Chromium(大多数人不使用)。 好的,我改了标题。 【参考方案1】:

警告:这是非常旧版本的 Safari 和 Chrome 的兼容性代码。现代浏览器都支持 FileReader API;这是一个教程:https://developer.mozilla.org/en-US/docs/Using_files_from_web_applications

此代码现在仅在出于某种原因需要支持 Safari 5 及更早版本或 Chrome 6 及更早版本时才有用。


一种可能是使用the method used in SwellJS:

像这样使用<input type="file" multiple="multiple" />

<form method="post" enctype="multipart/form-data" id="uploadform">
  <input type="file" name="dragupload[]" multiple="multiple"
  onchange="if (this.value) document.getElementById('uploadform').submit();" />
</form>

可以将输入元素设置为具有opacity: 0 的样式,并(绝对)定位在接受上传的元素上。整个表单可以放在iframe 中,用于类似“伪 Ajax”的行为。上传元素可以是隐藏的层,直到有东西被拖到它上面。

这样的 iframe 看起来像:

<script>
<!--
  var entered = 0;
-->
</script>
<body ondragenter="entered++;document.getElementById('uploadelement').style.display='block'" ondragleave="entered--;if (!entered) document.getElementById('uploadelement').style.display='none'">
  <form method="post" enctype="multipart/form-data" id="uploadform">
    Things can be dragged and dropped here!
    <input type="file" id="uploadelement" name="dragupload[]" multiple="multiple" onchange="if (this.value)  document.getElementById('uploadform').submit(); " style="display:none;position:absolute;top:0;left:0;right:0;bottom:0;opacity:0;" />
  </form>
</body>

只有在检测到 Safari 或 Chrome 时才应该这样做(因为其他浏览器不支持拖放到 &lt;input type="file" /&gt; 元素上),并且可以与 Firefox 3.6 的 HTML5 drop 事件结合使用+。

我不知道这是否是 Gmail 使用的方法,但它确实也可以。

【讨论】:

显示/隐藏代码中的条目计数递增/递减技巧非常方便。 感谢您的这篇文章。您是否可以从此处链接到完整示例?一个可能包括服务器端代码?或者只是解释(在 asp.net 上)我如何访问从这个表单(“uploadForm”)提交的文件【参考方案2】:

您可能对更符合技术和浏览器的东西感兴趣。

在我看来Plupload 做得很好,支持以下功能:

分块 拖放 PNG 调整大小 JPEG 调整大小 类型过滤 流上传 分段上传 文件大小限制 上传进度

适用于以下大多数技术:

闪光 齿轮 HTML 5 银光 BrowserPlus

是的,since 2010.05.27,它支持在 Chrome beta 上运行的 HTML5 拖放。

【讨论】:

哦?您是否尝试过此页面的“HTML 5 运行时”? plupload.com/example_all_runtimes.php HTML5 运行时版本适用于 OS X 上的 6.0.472.25 dev。它似乎也适用于 Firefox 4 测试版。 HTML5 上传也适用于我在 WinXp 上的 Chrome 8.552.210 Beta 上(仅几个月,他们有 2 个版本)。无论如何感谢您的帖子! 对我来说:在 html5 运行时,FF 15.0.1 有效; Chrome 21 没有。【参考方案3】:

经过大量、大量、大量的侦探工作,我在 Chrome 中找到了一些东西。这适用于 Chrome。在 Safari 上,它会冻结。在 Firefox 上,它不会让我删除文件。 IE 会打开删除的文件。即使在 Chrome 中,由于某种原因,拖放也只能工作一次,之后您必须刷新页面。 (一个可能的原因是事件处理程序有问题。)

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
    <head>
        <script type="text/javascript">
            window.onload = function () 
                var div = document.getElementById('div');
                div.ondragenter = div.ondragover = function (e) 
                    e.preventDefault();
                    e.dataTransfer.dropEffect = 'copy';
                    return false;
                
                div.ondrop = function (e) 
                    for (var i = 0; i < e.dataTransfer.files.length; i++)  // e.dataTransfer is a DataTransfer object (https://developer.mozilla.org/En/DragDrop/DataTransfer), e.dataTransfer.files is a FileList object (https://developer.mozilla.org/en/DOM/FileList)
                        var file = e.dataTransfer.files[i]; // file is a File object (https://developer.mozilla.org/en/DOM/File)

                        var xhr = new XMLHttpRequest;
                        xhr.open('post', 'handler.php', true);
                        xhr.onreadystatechange = function () 
                            if (this.readyState != 4)
                                return;
                            document.body.innerHTML += '<pre>' + this.responseText + '</pre>';
                        
                        xhr.setRequestHeader('Content-Type', 'multipart/form-data');
                        xhr.setRequestHeader('X-File-Name', file.fileName);
                        xhr.setRequestHeader('X-File-Size', file.fileSize);
                        xhr.send(file); // For some reason sending the actual File object in Chrome works?
                    

                    e.preventDefault();
                    return false;
                
            
        </script>
    </head>
    <body>
        <div id="div" style="width: 100%; height: 200px; border: 1px solid blue">Drop here</div>
    </body>
</html>

handler.php:

    // This is not a true file upload. Instead, it sends the raw data directly.
    echo htmlentities(file_get_contents('php://input'));

【讨论】:

使用 document.body.innerHTML += 会破坏 dom 事件。避免这种情况将解决您的问题。 效果很好,一个 gmail 开发人员告诉我,你应该对 FF3.6+ 使用 dropEffect='move',还没有在 FF 中测试过,所以不确定这是否仍然需要【参考方案4】:

您不需要使用 iframe 来进行伪 ajax 上传。 Chrome 和 Safari 都支持 XHR2 uploads 和进度事件,所以你可以做进度条等。

【讨论】:

...是的,这很明显。这里的问题是我见过的所有教程都使用FileReadergetAsBinary()(即Chrome 不支持的功能)来XHR2 上传文件。我怀疑这是必需的,所以我在这里基本上要做的是询问如果想要 Chrome 支持该怎么做。【参考方案5】:

对于我们自己的应用程序,我们只对 FireFox 进行拖放。我们恢复为其他人的传统 iframe 上传。为了检测是否支持拖放,我们运行以下代码:

if (typeof(window.File) == 'object' && typeof(window.FileReader) == 'function' && typeof(window.FileList) == 'object') 
   // DnD is supported!

希望这对某些人有所帮助。

【讨论】:

您确实提到您仅对 FireFox 进行拖放,但对于可能使用此代码的其他人,Safari 5 (Windows) 将为 window.FileReader 检查返回 false,即使它处理文件滴得很好。仍在为文件拖放寻找可靠的浏览器功能检测...【参考方案6】:

您可以使用 html5uploader 库:http://code.google.com/p/html5uploader/

它适用于 Firefox、Safari 和 Chrome。

【讨论】:

【参考方案7】:

最新浏览器支持文件上传。 你可以使用:

xhr = new XMLHttpRequest();     
xhr.open('POST', targetPHP, true);
var formData = new FormData();
formData.append('upload',file);
xhr.send(formData);

你不需要设置边界或任何头部,这样就可以了。 我在 客户端:firefox 6.02 和 chrome 13。 服务器:带有“spring mvc”的tomcat

【讨论】:

【参考方案8】:

您可以使用 FormData 来存储文件,然后上传。例如

function setUp()
  var dropContainer = document.getElementById("container");
  dropContainer.addEventListener("drop",dropHandler,false);
  dropContainer.addEventListener("dragenter", function(event)event.stopPropagation();event.preventDefault();, false);
  dropContainer.addEventListener("dragover", function(event)event.stopPropagation();event.preventDefault();, false);
  dropContainer.addEventListener("drop", dropHandler, false);
  getResult()

function dropHandler(event)
  var files = event.dataTransfer.files;
  var count = files.length;
  form = new FormData();
  for(var i= 0;i<count;i++)
    form.append("file"+i, files[i]);
  
  sendData();

function sendData()
  var xhr = new XMLHttpRequest();  
  xhr.upload.addEventListener("progress", uploadProgress, false);  
  xhr.addEventListener("load", uploadComplete, false);
  xhr.addEventListener("error", uploadFailed, false);  
  xhr.open("POST", "/upload");
  xhr.send(form);
  var progressBar = document.getElementById('progressBar');
  progressBar.style.display = 'block';
  progressBar.style.width = '0px';

演示在这里(http://flexinnerp.appspot.com/) 尽情享受吧:)

【讨论】:

【参考方案9】:

设置多个属性如:

input type="file" name="file1" multiple="multiple" class="DropHere"

并使用这个 CSS DropHere 类:

.DropHere

    height: 100px;
    padding: 3px;
    border: 2px dashed #555;
    border-radius: 5px;
    cursor: default;
    background-image:url("data:image/svg+xml;utf8, <svg xmlns='http://www.w3.org/2000/svg' version='1.1' height='100px' width='220px'><text x='55' y='75' font-size='20'>or drop files here</text></svg>");
    background-repeat: no-repeat;

文件字段现在看起来像:

如果你使用asp.net你可能也会喜欢我写的这篇文章“带进度条和拖放的多文件上传”:http://www.codeproject.com/Articles/818561/Multiple-file-upload-with-progress-bar-and-drag-an

【讨论】:

以上是关于在 Google Chrome/Chromium 和 Safari 中拖放文件上传?的主要内容,如果未能解决你的问题,请参考以下文章

Chrome/Chromium的实验性功能+扩展推荐,让你的Chrome/Chromium起飞!

Chrome/Chromium 中奇怪的对角线(错误?)

有没有办法在 javascript 中检测 Chrome/Chromium/V8 分析器是不是正在运行?

Chrome / Chromium为存在的页面提供404

如何使用更改和删除事件在 firefox 和 chrome/chromium 上上传和列出目录

Windows 上的 HTML5 地理位置比 Linux(Firefox、Chrome、[Chromium])更准确