jQuery插件之路——文件上传(支持拖拽上传)

Posted 我要心态好好哒

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了jQuery插件之路——文件上传(支持拖拽上传)相关的知识,希望对你有一定的参考价值。

 好了,这次咱一改往日的作风,就不多说废话了,哈哈。先贴上源代码地址,点击获取。然后直接进入主题啦,当然,如果你觉得我有哪里写的不对或者欠妥的地方,欢迎留言指出。在附上一些代码之前,我们还是先来了解下,上传文件的时候需要利用的一些必要的知识。

首先我们要说的就是FileReader对象,这是一个html5提出的,专门用来异步的读取用户计算机上文件的对象,这里有详细的介绍。所以如果我们想要使用它,那么首先我们得先创建一个FileReader对象。

var fr = new FileReader()

1、这个对象拥有五个方法:

方法名

参数

描述

abort

中断读取

 readAsArrayBuffer

file

开始读取file中的内容

readAsBinaryString

file 将文件读取为二进制码
readAsDataUrl file 将文件读取为URL
readAsText file,[encoding] 讲文件读取为文本

下面附上一个例子:

<input type="file" id="file"/>
    <img src="" alt="" id="img">
    <script src="jquery.min.js"></script>
    <script>
        var ipt = $(\'#file\'),
        img = $(\'#img\');
        ipt.change(function () {
            var fr = new FileReader();
            fr.readAsDataURL(this.files[0]);
            fr.onload = function () {
                img.attr(\'src\', fr.result);
            }
        })
    </script>

效果图:,其他的几个方法也基本上大同小异,所以在这里就不做过多解释了。

2、这个对象还拥有三个状态常量:

常量名 描述
EMPTY 0 还没有加载任何数据
LOADING 1 数据正在被加载
DONE 2 加载完毕

3、这个对象还拥有三个属性:

属性名 类型 描述
error DOMError 在读取文件时发生的错误. 只读.
readyState unsigned short 表明FileReader对象的当前状态. 值为State_constants中的一个. 只读
result jsval 读取到的文件内容.这个属性只在读取操作完成之后才有效,并且数据的格式取决于读取操作是由哪个方法发起的. 只读.

4、6个事件处理程序:

事件名 描述
onabort 当读取操作被中止时调用
onerror 当读取操作发生错误时调用.
onload 当读取操作成功完成时调用.
onloadend 当读取操作完成时调用,不管是成功还是失败.该处理程序在onload或者onerror之后调用.
onloadstart 当读取操作将要开始之前调用
onprogress 在读取数据过程中周期性调用.

这里我们再来说说formData对象,同样的我们利用它来上传文件,首先需要创建一个formData对象实例

var formData = new FormData();

这个对象有一个append方法,该方法接受三个参数:name、value、filename

参数名 描述
name   字段名称
value 字段值.可以是,或者一个字符串,如果全都不是,则该值会被自动转换成字符串.
filename 可选,文件名。

在使用这个对象上传文件的时候,我们需要注意一点,需要在form标签上添加上enctype="multipart/form-data"这个属性,用来设置表单的MIME编码,因为默认的编码格式是application /x-www-form-urlencoded,不能用于文件上传,也可以在使用jQuery的$.ajax方法的时候,设置data属性为formData。

上面就是该DEMO主要用到的知识点,下面附上一些源代码,和效果图。

HTML代码:

<div class="up_load_file">
    </div>
    <script src="js/jquery-1.11.3.js"></script>
    <script src="js/uploadfile.js"></script>
    <script>
        $(\'.up_load_file\').uploadfile({
            url : \'file.php\',
            width : 500,
            height : 50,
            canDrag : true,
            canMultiple : true,
            success: function (fileName) {
                alert(fileName + \'上传成功\');
            },
            error: function (fileName) {
                alert(fileName + \'上传失败\');
            },
            complete : function () {
                alert(\'所有文件上传完毕\');
            }
        });
    </script>

JS代码:

;(function ($, undefined) {
    $.fn.uploadfile = function (setting) {
        var defaultSetting = {
            url : \'file.php\',
            width : 600,
            height : 50,
            canDrag : true,
            canMultiple : true,
            success : function (fileName) {   //单个文件上传成功的回调函数
            },
            error : function (fileName) {     //单个文件上传失败的回调函数
            },
            complete : function () {  //上传完成的回调函数
            }
        };

        //判断浏览器是否支持FileReader
        if(!window.FileReader){
            alert(\'您的浏览器不支持FileReader,请更换浏览器。\');
            return;
        }

        setting = $.extend(true, {}, defaultSetting, setting);
        setting.width < 450 && (setting.width = 450);

        $(this).each(function (i, item) {
            var demoHtml = \'\';
            //是否可以拖拽图片上传,构造dom结构
            if(setting.canDrag){
                setting.height < 200 && (setting.height = 200);
                demoHtml +=              \'<div class="file_sel">\';
                demoHtml +=                    \'<div class="file_input">\';
                demoHtml +=                        \'<div class="sel_file_img">\';
                demoHtml +=                            \'<span><img src="img/add_img.png"/></span>\';
                demoHtml +=                        \'</div>\';
                demoHtml +=                        \'<div class="sel_file_btn">\';
                demoHtml +=                            \'<input type="file"/>\';
                demoHtml +=                            \'<button>点击选择文件</button>\';
                demoHtml +=                        \'</div>\';
                demoHtml +=                    \'</div>\';
                demoHtml +=                    \'<div class="file_drag">\';
                demoHtml +=                        \'<span>或者将文件拖到此处</span>\';
                demoHtml +=                    \'</div>\';
                demoHtml +=                \'</div>\';
                demoHtml +=                \'<div class="file_info_handle">\';
                demoHtml +=                    \'<div class="file_info">\';
                demoHtml +=                        \'当前选择了<span class="file_count">0</span>个文件,共<span class="file_size">0</span>KB。\';
                demoHtml +=                        \'<input type="file"/>\';
                demoHtml +=                        \'<button class="continue_sel">继续选择</button>\';
                demoHtml +=                        \'<button class="uploadfile">开始上传</button>\';
                demoHtml +=                    \'</div>\';
                demoHtml +=                \'</div>\';
                demoHtml +=                \'<div class="file_show">\';
                demoHtml +=                \'</div>\';
            }else{
                setting.height < 50 && (setting.height = 50);
                $(item).addClass(\'noDrag\');
                demoHtml +=        \'<div class="file_info_handle">\';
                demoHtml +=            \'<div class="file_info">\';
                demoHtml +=                \'当前选择了<span class="file_count">0</span>个文件,共<span class="file_size">0</span>KB。\';
                demoHtml +=                \'<input type="file"/>\';
                demoHtml +=                \'<button class="continue_sel">继续选择</button>\';
                demoHtml +=                \'<button class="uploadfile">开始上传</button>\';
                demoHtml +=            \'</div>\';
                demoHtml +=        \'</div>\';
                demoHtml +=        \'<div class="file_show">\';
                demoHtml +=            \'<div class="sel_file_btn">\';
                demoHtml +=                \'<input type="file"/>\';
                demoHtml +=                \'<div class="sel_btn"></div>\';
                demoHtml +=            \'</div>\';
                demoHtml +=        \'</div>\';
            }
            $(item).css({
                width : setting.width,
                height : setting.height,
                display : \'block\'
            });
            $(item).html(demoHtml);

            //获取DOM节点
            var fileArr = [],
            fileSize = 0,
            _this = $(item),
            fileDrag = $(\'.file_sel .file_drag\', _this),
            selFileIpt = $(\'input[type=file]\', _this),
            selFileBtn = selFileIpt.next();
            fileCount = $(\'.file_info_handle .file_info .file_count\', _this),
            fileSz = $(\'.file_info_handle .file_info .file_size\', _this),
            beginUpload = $(\'.file_info_handle .file_info .uploadfile\', _this),
            fileShow = $(\'.file_show\', _this),
            noDragSelFile = $(\'.file_show .sel_file_btn\', _this);
            
            //显示拖拽上传部分
            setting.canDrag || fileShow.show();

            //是否可以多选
            setting.canMultiple && selFileIpt.attr(\'multiple\', \'multiple\');

            //绑定事件
            selFileIpt.on(\'change\', selFile);

            //让按钮去触发input的click事件
            selFileBtn.on(\'click\', function () {  
                $(this).prev().click();
            })

            fileDrag.on({
                dragover : dragOver, 
                drop : selFile
            })

            beginUpload.on(\'click\', upLoadFile);

            

            // 选择文件
            function selFile (e) {
                e = e || window.event;
                //阻止浏览器的默认行为
                if(e.preventDefault){  
                    e.preventDefault();    
                }else{
                    e.returnValue = false;
                }
                var files = this.files || event.dataTransfer.files,
                src = \'img/\',
                imgSrc;
                Array.prototype.forEach.call(files, function (item, i) {

                    //防止重复选择相同的文件
                    var notExist = fileArr.some(function (existFile) {
                        return existFile.name === item.name;
                    })
                    if(notExist && fileArr.length != 0){
                        return !notExist;
                    }

                    fileArr.push(item);
                    var fr = new FileReader();
                    fr.readAsDataURL(item);
                    fr.onload = function () {

                        //判断展示的文件类型
                        if(item.type.indexOf("image") > -1){
                            imgSrc = fr.result;
                        }else if(item.name.indexOf("rar") > -1){
                            imgSrc = src + \'rar.png\';
                        }else if(item.name.indexOf("zip") > -1){
                            imgSrc = src + \'zip.png\';
                        }else if(item.type.indexOf("text") > -1){
                            imgSrc = src + \'txt.png\';
                        }else{
                            imgSrc = src + \'file.png\';
                        }

                        //展示选择的文件
                        var imgDom = $(\'<span class="img_box"><span class="up_load_success" title="上传成功"></span><span class="img_handle"><span class="file_name" title="\'+ item.name +\'">\'+ item.name +\'</span><span class="icon-bin"></span></span><img src="\'+ imgSrc +\'"/></span>\');
                        if(setting.canDrag){
                            fileShow.css(\'display\') === \'none\' && fileShow.show();
                            fileShow.append(imgDom);
                        }else{
                            fileShow.css(\'display\') === \'none\' && fileShow.show();
                            noDragSelFile.before(imgDom);
                        }
                    }
                }) 

                //选择的文件的信息
                fileCount.html(fileArr.length);
                fileSz.html(getFileInfo());

                //防止在删除了上次选择的文件后,再次选择相同的文件无效的问题。
                this.value =\'\';  
            }

            //拖拽
            function dragOver (e) {
                var event = e || window.event;
                event.preventDefault();
            }

            //上传文件
            function upLoadFile () {
                if(!fileArr.length){
                    alert(\'请选择文件\');
                    return;
                }
                fileArr.forEach(function (item, i) {
                    var upLoadSuccess = $(\'.img_box\').eq(i).children(\'.up_load_success\');
                    
                    //防止重复上传
                    if(upLoadSuccess.css(\'display\') === \'block\') return false;   
                    var formData = new FormData();
                    formData.append(\'file\', item);
                    $.ajax({
                        url: setting.url,
                        type: \'POST\',
                        cache: false,
                        data: formData,
                        processData: false,
                        contentType: false
                    }).done(function(res) {
                        //上传成功图标
                        upLoadSuccess.show();

                        //单个文件上传成功执行回调
                        setting.success(item.name);

                        //全部文件上传完成执行回调函数
                        (i === (fileArr.length - 1)) && setting.complete();
                    }).fail(function(res) {
                        //单个文件上传失败执行回调
                        setting.error(item.name);

                        (i === (fileArr.length - 1)) && setting.complete();
                    });
                })
            }

            //计算文件信息
            function getFileInfo () {
                //每次重新计算大小,防止单位不同造成错误
                fileSize = 0;
                fileArr.forEach(function (item, i) {
                    fileSize += item.size;
                })
                fileSize = (fileSize / 1024).toFixed(2);
                return fileSize;
            }

            fileShow.on(\'click\', \'.icon-bin\' , function () {
                //删除节点
                var index = $(this).parents(\'.img_box\').index();
                $(this).parents(\'.img_box\').remove();

                //删除上传文件
                fileArr.splice(index, 1);

                //修改文件信息
                fileCount.html(fileArr.length);
                fileSz.html(getFileInfo());

                //隐藏文件显示区域
                !setting.canDrag || fileArr.length || fileShow.hide();
            })
        })
    }
})(jQuery)
View Code

后台PHP代码:

$fileName = $_FILES[\'file\'][\'name\'];
    $type = $_FILES[\'file\'][\'type\'];
    $size = $_FILES[\'file\'][\'size\'];
    $fileAlias = $_FILES["file"]["tmp_name"];

    if($fileAlias){
        move_uploaded_file($fileAlias, "uploadfile/" . $fileName);
    }
    echo \'fileName: \' . $fileName . \', fileType: \' . $type . \', fileSize: \' . ($size / 1024) . \'KB\';

 

支持拖拽上传样式。不支持拖拽的样式。

代码中一些必要的地方已经写好注释了,这里也就不做过多解释,今天就先写到这里了。作者敲代码也不容易,如果你觉得这些内容还有那么一些价值的话,请点下赞,谢谢^_^。

 

以上是关于jQuery插件之路——文件上传(支持拖拽上传)的主要内容,如果未能解决你的问题,请参考以下文章

jQuery文件上传插件jQuery Upload File 有上传进度条

前端开发:功能强大的jQuery文件上传插件(jQuery File Upload)

哪个javascript框架支持ajax方式的文件上传

在线文件拖拽上传插件——DropzoneJS

基于bootstrap的上传插件fileinput实现ajax异步上传功能(支持多文件上传预览拖拽)

基于 jq 实现拖拽上传 APK 文件,js解析 APK 信息