在这段 JSP 代码中,我们从 POST 请求中接受文件名字以及二进制数据。将二进制数据写入到服务器的“D:\”路径中。并返回文件的完整路径。以上代码可以在最新的 Firefox 3.6 中调试通过。
四、使用拖拽上传文件
前面我们介绍了怎样通过 HTML5 的 File API 来读取本地文件内容并上传到服务器,通过这种方式已经能够满足大部分用户的需求了。其中一个不足是用户只能通过点击“浏览”按钮来逐个添加文件,如果需要批量上传文件,会导致用户体验不是很友好。而在桌面应用中,用户一般可以通过鼠标拖拽的方式方便地上传文件。拖拽一直是 Web 应用的一个软肋,一般浏览器都不提供对拖拽的支持。虽然 Web 程序员可以通过鼠标的 mouseenter,mouseover 和 mouseout 等事件来实现拖拽效果,但是这种方式也只能使拖拽的范围局限在浏览器里面。一个好消息是 HTML5 里面不仅加入了 File API,而且加入了对拖拽的支持,Firefox 3.5 开始已经对 File API 和拖拽提供了支持。下面我们先简要介绍一下拖拽的使用,然后用一个例子来说明如何通过拖拽上传文件。
1、拖拽简介
拖拽一般涉及两个对象:拖拽源和拖拽目标。
拖拽源:在 HTML5 草案里如果一个对象可以作为源被拖拽,需要设置 draggable 属性为 true 来标识该对象可作为拖拽源。然后侦听源对象的 dragstart 事件,在事件处理函数里设置好 DataTransfer。在 DataTransfer 里可以设置拖拽数据的类型和值。比如是纯文本的值,可以设置类型为"text/plain",url 则把类型设置为"text/uri-list"。 这样,目标对象就可以根据期望的类型来选择数据了。
拖拽目标:一个拖拽目标必须侦听 3 个事件。
dragenter:目标对象通过响应这个事件来确定是否接收拖拽。如果接收则需要取消这个事件,停止时间的继续传播。
dragover:通过响应这个事件来显示拖拽的提示效果。
drop:目标对象通过响应这个事件来处理拖拽数据。在下面的例子里我们将在 drop 事件的处理函数里获取 DataTransfer 对象,取出要上传的文件。
由于本文主要介绍 File API,对这部分不作详细解释,感兴趣的读者可以参考 HTML5 草案(见参考资料)。
2、拖拽上传文件实例
下面以一个较为具体的例子说明如何结合拖拽和 File API 来上传文档。由于直接和桌面交互,所以我们不需要处理拖拽源,直接在目标对象里从 DataTransfer 对象获取数据即可。
首先,我们需要创建一个目标容器用来接收拖拽事件,添加一个 div 元素即可。然后用一个列表来展示上传文件的缩略图,进度条及文件名。参见清单 8 的 HTML 代码和图 3 的效果图。详细代码请参见附件中的 dnd.html 文件。
清单 8 拖曳目标的 HTML 代码
<div> <span>Drag and drop files here to upload.</span> <ul></ul> </div>
拖拽目标创建好之后,我们需要侦听其对应的事件 dragenter,dragover 和 drop。在 dragenter 事件处理函数里,我们只是简单地清除文件列表,然后取消 dragenter 事件的传播,表示我们接收该事件。更加妥当的作法是判断 DataTransfer 里的数据是否为文件,这里我们假设所有拖拽源都是文件。dragover 事件里我们取消该事件,使用默认的拖拽显示效果。在 drop 事件里我们注册了 handleDrop 事件处理函数来获取文件信息并上传文件。清单 9 展示了这些事件处理函数。
清单 9 设置事件处理函数
function addDNDListeners() { var container = document.getElementById("container"); var fileList = document.getElementById("fileList"); // 拖拽进入目标对象时触发 container.addEventListener("dragenter", function(event) { fileList.innerHTML =''; event.stopPropagation(); event.preventDefault(); }, false); // 拖拽在目标对象上时触发 container.addEventListener("dragover", function(event) { event.stopPropagation(); event.preventDefault(); }, false); // 拖拽结束时触发 container.addEventListener("drop", handleDrop, false); } window.addEventListener("load", addDNDListeners, false);