关于js拖拽上传 [一个拖拽上传修改头像的流程]


Posted in Javascript onJuly 13, 2011

如今现代的浏览器已经有很多支持拖拽文件读取操作,其优点不再复述。前端时间利用拖拽改进了一下网站的头像上传流程,对其中的要点和实践体会做一点总结。

 先看一下总体视图:
关于js拖拽上传 [一个拖拽上传修改头像的流程]
1、 文件拖拽接受区域要有明显的标示,并且要尽可能的大(由于版面的原因,这个界面的拖放盒子并不大)。可以用虚线框盒子等样式吸引用户拖拽文件。最好有明显的文字提示和图标配合。

2、 在交互体验上当文件拖入浏览器窗口时,可以用拖放区变换背景颜色等向用户发起放置操作邀请。
实现代码:

doc.bind({ 
'dragenter':function(e){ 
$("#brsbox").addClass("dragbrowse"); 
dropbox.addClass("shine"); 
return false; 
}, 
'dragleave':function(e){ 
dropbox.removeClass("shine"); 
return false; 
}, 
'drop':function(e){ 
stopdft(e);} 
}); 
dropbox.bind({ 
'dragenter':function(e){ 
dropbox.addClass("candrop"); 
stopdft(e);}, 
'dragleave':function(e){ 
dropbox.removeClass("candrop"); 
stopdft(e);}, 
'dragover':function(e){ 
stopdft(e);}, 
'drop':function(e){ }

对于不支持拖拽的浏览器:

可惜的是 某些浏览器并不支持文件拖拽读取,这其中包括IE9等较现代的浏览器。所以我们必须为不支持拖拽的浏览器准备普通文件浏览上传作为备用方案。
当不支持拖拽文件读取时,界面如下:

 关于js拖拽上传 [一个拖拽上传修改头像的流程]

实现检测的代码如下:

it.detectDragable = function(){ 
filedrag = !!window.FileReader; 
if(!filedrag) return; 
$("#avtcnt").addClass('dragable');

文件放置时的处理:

 

 关于js拖拽上传 [一个拖拽上传修改头像的流程]

 文件放置到可接受区域时,请注意这时候无论你拖放在鼠标上的文件是单个还是多个,在浏览器和操作系统之间传送的e.dataTransefer.files总是复数。也就是多个文件。这也就意味着你需要循环处理鼠标上所携带的文件。
代码如下:

dropdom.addEventListener('drop',function(e){ 
it.handlefile(e.dataTransfer.files); 
stopdft(e);},false); }; 
it.handlefile = function(files){ 
var noimg = 0; 
for(var i=0; i<files.length; i++){ 
var file = files[i]; 
if(!file.type.match(/image*/)){ 
noimg ++; 
if(noimg ==files.length){ 
QSL.optTips('请选择jpg, png, gif 等格式的图片'); 
return false; 
} 
continue; 
} 
var reader = new FileReader(); 
reader.onload = function(e){ 
var img = document.createElement('img'); 
img.src = reader.result; 
setTimeout(function(){ 
it.imgSize = { 
w:img.width, 
h:img.height 
}; 
},500); 
dropdom.innerHTML=""; 
img.className ='localimg'; 
it.imgData = reader.result; 
dropdom.appendChild(img); 
imagedata.empty().val(reader.result); 
dropbox.addClass("droped"); 
clearner.show(); 
}; 
reader.readAsDataURL(file); 
}

处理拖拽到浏览器的文件

其中 stopdft(e) 是为了防止浏览器默认操作,不以浏览器打开文件。而转由脚本来处理拖放的文件。
这个流程中,我们需要的是图片文件,所以便利操作 e.dataTransfer.files 对象,查找类型为image的文件。
如果没有,则会提示。

读取文件的关键代码:
var reader = new FileReader();

reader.onload = function(e){
var img = document.createElement('img');
img.src = reader.result;
};
reader.readAsDataURL(file);
本例中我们需要读取图片的高度和宽度属性。所以我们做了如下操作
setTimeout(function(){
it.imgSize = {
w:img.width,
h:img.height
};
},500);
虽然是本地文件读取,但是仍然要延时来保证图片确实读取完毕。否则在某些浏览器中会取不到宽高的值。(可否有其他更简便方法?望指出)

删除现有图片,重置拖拽区域:
浏览读取完本地图片之后,要给用户提供删除和重置的选项。(如果是直接上传当然更简便)

it.resetDropbox = function(){ 
dropbox.attr("class","dropbox") 
.empty() 
.text("将文件拖拽至此区域"); 
imgData = ''; 
it.imgData = ''; 
it.imgSize = {w:0,h:0}; 
picsub.removeClass("uploading") 
.find("button").removeAttr("disabled") 
.text("上传"); 
imagedata.val(''); 
clearner.hide();

重置拖拽区域

到这里拖放读取文件的流程基本结束。
利用拖放,读取本地文件的其他优点:
普通的上传更改图片流程是:选择图片-上传图片-上传成功-服务器返回图片-客户端浏览效果
而如果利用拖放读取本地文件则可省去服务器返回 图片的步骤,直接利用reader.result返回的数据。
这样就节省了从服务器读取图片的延迟,并且节省了往返的数据流量。所以只需确认服务器端图片上传成功,图片预览调取本地数据即可:
代码:

function initImageCrop(url){ 
var t = document.getElementById("target"), 
p = document.getElementById("preview"), 
b = browseImage, 
s = [], 
ts = []; 
if(url=='data'){ 
t.src = b.imgData; 
p.src = b.imgData; 
posImage(b.imgSize.w,b.imgSize.h); 
}else{ 
var cutimg = new Image(); 
cutimg.onload = function(){ 
t.src = url; 
p.src = url; 
posImage(cutimg.width,cutimg.height); 
} 
cutimg.src = url;

图片上传成功后的处理

完整DEMO预览(静态文件暂时没有上传成功后的展示(:)
DEMO脚本

Javascript 相关文章推荐
jQuery使用prepend()方法在元素前添加内容用法实例
Mar 26 Javascript
apply和call方法定义及apply和call方法的区别
Nov 15 Javascript
JS操作XML实例总结(加载与解析XML文件、字符串)
Dec 08 Javascript
Bootstrap面板(Panels)的简单实现代码
Mar 17 Javascript
node打造微信个人号机器人的方法示例
Apr 26 Javascript
详解基于mpvue的小程序markdown适配解决方案
May 08 Javascript
vue devtools的安装与使用教程
Aug 08 Javascript
RequireJS用法简单示例
Aug 20 Javascript
微信小程序实现星星评价效果
Nov 02 Javascript
使用gulp构建前端自动化的方法示例
Dec 25 Javascript
浅谈Node新版本13.2.0正式支持ES Modules特性
Nov 25 Javascript
详解ES6数组方法find()、findIndex()的总结
May 12 Javascript
使用 JScript 创建 .exe 或 .dll 文件的方法
Jul 13 #Javascript
JS模拟面向对象全解(二、类型与赋值)
Jul 13 #Javascript
JS模拟面向对象全解(一、类型及传递)
Jul 13 #Javascript
一些实用的jQuery代码片段收集
Jul 12 #Javascript
formValidator3.3的ajaxValidator一些异常分析
Jul 12 #Javascript
在IE浏览器中resize事件执行多次的解决方法
Jul 12 #Javascript
JQuery获取当前屏幕的高度宽度的实现代码
Jul 12 #Javascript
You might like
蝙蝠侠:侠影之谜
2020/03/04 欧美动漫
曾在DC漫画界反派角色扮演的演员,谁才是你心目中的小丑之王?
2020/04/09 欧美动漫
PHP去除数组中重复的元素并按键名排序函数
2008/08/18 PHP
PHP中spl_autoload_register()和__autoload()区别分析
2014/05/10 PHP
php相对当前文件include其它文件的方法
2015/03/13 PHP
php将图片保存为不同尺寸图片的图片类实例
2015/03/30 PHP
php ci 获取表单中多个同名input元素值的代码
2016/03/25 PHP
PHP微信开发之查询城市天气
2016/06/23 PHP
PHP切割汉字的常用方法实例总结
2019/04/27 PHP
js 函数的执行环境和作用域链的深入解析
2009/11/01 Javascript
JavaScript实现拼音排序的方法
2012/11/20 Javascript
js 获取和设置css3 属性值的实现方法
2013/05/06 Javascript
[JSF]使用DataModel处理表行事件的实例代码
2013/08/05 Javascript
Eclipse去除js(JavaScript)验证错误
2014/02/11 Javascript
深入浅析jQuery对象$.html
2016/08/22 Javascript
js 文字超出长度用省略号代替,鼠标悬停并以悬浮框显示实例
2016/12/06 Javascript
jQuery中可见性过滤器简单用法示例
2018/03/31 jQuery
Angular CLI在Angular项目中如何使用scss详解
2018/04/10 Javascript
Vue.js实现的购物车功能详解
2019/01/27 Javascript
vue基于两个计算属性实现选中和全选功能示例
2019/02/08 Javascript
微信小程序利用云函数获取手机号码
2019/12/17 Javascript
Vue使用轮询定时发送请求代码
2020/08/10 Javascript
[04:05]TI9战队采访 - Natus Vincere
2019/08/22 DOTA
python实现根据用户输入从电影网站获取影片信息的方法
2015/04/07 Python
Python中使用Queue和Condition进行线程同步的方法
2016/01/19 Python
python开发之str.format()用法实例分析
2016/02/22 Python
Python面向对象编程基础解析(二)
2017/10/26 Python
Python+PyQt5实现美剧爬虫可视工具的方法
2019/04/25 Python
使用Python Pandas处理亿级数据的方法
2019/06/24 Python
python里dict变成list实例方法
2019/06/26 Python
使用TensorBoard进行超参数优化的实现
2020/07/06 Python
HTML5在canvas中绘制复杂形状附效果截图
2014/06/23 HTML / CSS
校运会入场式解说词
2014/02/10 职场文书
2015暑期工社会实践报告
2015/07/13 职场文书
详解Python requests模块
2021/06/21 Python
MySQL如何使备份得数据保持一致
2022/05/02 MySQL