Fine Uploader文件上传组件应用介绍


Posted in PHP onJanuary 06, 2013

最近在处理后台数据时需要实现文件上传.考虑到对浏览器适配上采用Fine Uploader. Fine Uploader 采用ajax方式实现对文件上传.同时在浏览器中直接支持文件拖拽[对浏览器版本有要求类似IE版本必须是9或是更高的IE10].在不同浏览器中提供统一用户体验.该组件基本覆盖目前所有主流浏览器.同时没有任何第三方组件依赖.相当Clear.在服务器端已经覆盖支持了ASP.NET/ColdFusion/Java/Node.js/Perl/PHP/Python. 对上传细节类似限制文件大小,文件类型,文件上传的数量等通过统一接口以暴露选项方式操作.

看到Github 上Fine Uploader 按照官方的说法. Fine Uploader前身是Ajax Upload. 新版本Fine Uploader主要添加一些新特性.从1.0版本发布的Realse Note来看.二者最大的区别在于.Fine Uploder不在基于Jquery组件.而某些细节处理也更加统一严格.类似返回值全部统一为Json格式.对后台服务器操作和前端Dom对象一些操作Code全部集中Js Script脚本文件中.这样集成使Fine Uploader组件使用非常简单.只需要添加一个CSS+JavaScript文件即可实现文件上传.大大简化用户引用和操作组件难度.

Fine Uploader特点如下:
Fine Uploader Features:
A:支持文件上传进度显示.
B:文件拖拽浏览器上传方式
C:Ajax页面无刷新.
D:多文件上传.
F:跨浏览器.
E:跨后台服务器端语言.
在Git Hub上Fine Uploader上下载打包源码,在Php Designer 8中打开其源码可以看到其源码结构如下:
Fine Uploader文件上传组件应用介绍 
在根目录中可以看到Client客户端调用需要使用文件.Server目录则是对应不同语言Perl/Php/Asp.net[VB]等版本实现.test目录则有包含一个完整本地Sample Demo.可供参考.
如何快速构建一个简单Demo? 其实官方在Basic-Demo-Page上已经给出一个简单的演示.这里基于Bootstrap方式构建.
首先新建一个Html空白页面.命名FineUploderDemo.html.添加如下CSS引用如下:

<link href="static/css/fineuploader.css" rel="stylesheet"> 
<link href="static/css/bootstrap.min.css" rel="stylesheet">

这两个文件时必须引用的.fineuploader.css则是对应下载Fine Uploder源码Client目录下.fineuploder.css 提供JS脚本中所需的CSS样式,主要包括按钮的样式、进度显示的样式以及上传结果的样式.添加JavaScript文件引用如下:
<script src="static/script/fineupload/header.js"></script> 
<script src="static/script/fineupload/util.js"></script> 
<script src="static/script/fineupload/button.js"></script> 
<script src="static/script/fineupload/handler.base.js"></script> 
<script src="static/script/fineupload/handler.form.js"></script> 
<script src="static/script/fineupload/handler.xhr.js"></script> 
<script src="static/script/fineupload/uploader.basic.js"></script> 
<script src="static/script/fineupload/dnd.js"></script> 
<script src="static/script/fineupload/uploader.js"></script>

其中uploder.js和uploder.basic.js则是前端的所有上传功能都在该脚本中实现.必须引用.
同时添加client目录下processing和loading两张进度显示所需要的动态图片.该图片都在fineuploder.css文件调用.
在body添加如下Code:
<div id="bootstrapped-fine-uploader"></div> 
<script> 
function createUploader() { 
var uploader = new qq.FineUploader({ 
element: document.getElementById('bootstrapped-fine-uploader'), 
request: { 
endpoint: 'server/handlerfunction' 
}, 
text: { 
uploadButton: '<i class="icon-upload icon-white"></i> Click me now and upload a product image' 
}, 
template: 
'<div class="qq-uploader span12">' + 
'<pre class="qq-upload-drop-area span12"><span>{dragZoneText}</span></pre>' + 
'<div class="qq-upload-button btn btn-success" style="width: auto;">{uploadButtonText}</div>' + 
'<span class="qq-drop-processing"><span>{dropProcessingText}</span>'+ 
'<span class="qq-drop-processing-spinner"></span></span>' + 
'<ul class="qq-upload-list" style="margin-top: 10px; text-align: center;"></ul>' + 
'</div>', 
classes: { 
success: 'alert alert-success', 
fail: 'alert alert-error' 
}, 
debug: true 
}); 
} window.onload = createUploader; 
</script>

这是基于Bootstrap实现对Fine Uploader最简单的前端调用.前端一般需要做两件事A:添加Css+Js文件引用.B:在Js中实例化qq.FineUploder对象.运行效果如下:
Fine Uploader文件上传组件应用介绍 
查看JS构建qq.Fineuploader对象创建过程.首先指定Fine Uploader插件的Dom元素.通过Dom获取操作.request则是对应服务器端实现文件路径.在这建议不要自己构建服务器端处理.而是直接采用官方提供的实现文件修改即可.template则是对应上传文件添加内容模版也可以自己修改.debug是一个布尔值.用来控制是否使用浏览器的控制台打印Fine Uploader的调试信息,默认为false.
qq.FineUploader对象还有如下控制参数:
validation:该参数一般用来在执行上传文件操作前.在客户端做一些验证.验证操作包含文件格式.文件大小.等添加格式如下:
validation: 
{ 
allowedExtensions: ['jpeg', 'jpg', 'gif', 'png'], 
sizeLimit: 204800 // 200 kB = 200 * 1024 bytes 
} 
allowedExtensions控制上传文件的后缀格式数组. 
sizeLimit上传文件大小的上限,单位为byte的数值.浏览器不一定支持本设置.也可以在服务器端里设置. 
minSizeLimit:上传文件大小的下限,单位为byte的数值.同上有些浏览器存在适配问题.建议统一在服务端设置. 
另外针对qq.FineUploder对象在执行上传操作整个过程.定义了五个客户端可控做额外操作的事件.可以再callback参数下设置定义: 
callbacks: 
{ 
onSubmit: function(id, fileName) { 
$messages.append('<div id="file-' + id + '" class="alert" style="margin: 20px 0 0"></div>'); 
}, 
onUpload: function(id, fileName) { 
$('#file-' + id).addClass('alert-info') 
.html('<img src="client/loading.gif" alt="Initializing. Please hold."> ' + 
'Initializing ' + 
'“' + fileName + '”'); 
} 
}

onSubmit事件:文件开始提交.调用参数格式如下:onSubmit:function(id,fileName){}.
onUpload事件: 文件开始上传.调用参数格式如下:onUpload: function(id, fileName) {}.
onProgress事件: 文件正在上传.调用参数格式如下:onProgress:function(id,fileName,loaded,total){}.
onComplete事件: 文件上传成功. 调用参数格式如下:onComplete:function(id,fileName,responseJSON){}.
onCancel事件: 取消文件上传.调用参数格式如下:onCancel:function(id,fileName){}.
如上五个事件基本覆盖整个上传文件操作中所有过程.完全以开放的形式可以再客户端操作.关于调用如上事件参数说明如下:
Id:表示第几个开始上传的文件.Fine Uploder定义是默认从0开始计数.
fileName:上传文件的文件名.
loaded:表示已经上传到服务器端数据的大小[byte].
total: 需要上传文件的大小.
responseJSON: 用来在上传操作完成后返回的Json格式的数据.通过Jquery反序列化出来对象.其中包含一个IsSuccess属性用来判断此次上传是否成功.
如果你想在上传过程向服务器端传递数据.可以通过如下参数控制:
params:用来向服务器端传递数据.注意params采用key-value的数组存储.采用Post方式发送给服务器端.一般传递参数格式如下:
params: 
{ 
argument1: "value1", 
argument2: "value2" 
},

ok.这时基本关于Fine Uploader客户端初始化和控制操作选项基本完成.当我们需要上传操作时.如果IsAuto=false时可以通过已经定义qq.FineUploader对象的uploadStoreFiles()方式手工触发上传操作:
$('#triggerUpload').click(function() { 
uploader2.uploadStoredFiles(); 
});

如果我们此时点击上传会发现.则提示上传失败. 因为还没有对上传服务器端做任何处理:
request: 
{ 
endpoint: 'server/handlerfunction' 
},

这时我们需要在EndPoint指定处理文件上传的Php文件[这里是phpdemo].关于服务器端如果你没有已经成熟处理模块.还是推荐你使用官方Server目录上.这里我采用php环境则选中时php.php文件.对应客户端修改如下:
request: 
{ 
endpoint: 'controller/php.php' 
}

打开php.php发现在文件头部说明该文件使用同时在文件定义三个类用来分别处理XMLHttpRequest、FormPost、BasicPost方式文件服务器端处理.在文件顶部注释中:
/****************************************
Example of how to use this uploader class...
You can uncomment the following lines (minus the require) to use
hese as your defaults.

// list of valid extensions, ex. array("jpeg", "xml", "bmp")
$allowedExtensions = array();
// max file size in bytes
$sizeLimit = 10 * 1024 * 1024;
//the input name set in the javascript
$inputName = 'qqfile'

require('valums-file-uploader/server/php.php');
$uploader = new qqFileUploader($allowedExtensions, $sizeLimit, $inputName);

// Call handleUpload() with the name of the folder, relative to PHP's getcwd()
$result = $uploader->handleUpload('uploads/');

// to pass data through iframe you will need to encode all html tags
header("Content-Type: text/plain");
echo htmlspecialchars(json_encode($result), ENT_NOQUOTES);

/******************************************/
已经详细说明如下Class调用方式.添加如下Php代码即可简单完成服务器端处理:

$allowedExtensions = array("jpeg", "jpg", "bmp", "png"); 
$sizeLimit = 10 * 1024 * 1024; 
$uploader = new qqFileUploader($allowedExtensions, $sizeLimit); 
$result = $uploader->handleUpload('uploads/'); //folder for uploaded files 
echo htmlspecialchars(json_encode($result), ENT_NOQUOTES);

allowExtensions则定义了允许上传文件的格式.
sizeLimit上限定义为10M.注意首先采用Phpinfo();方法输出当前php环境配置.一般默认情况默认上传文件最大大小为2M.如果你需要上传更大则修改php.ini文件配置参数 这里不再赘述.
uploder则是初始化qq.Fileuploder对象.并加载配置.

fineuploder调用处理上传函数.并传递服务器端存储上传文件存储路径.
echo想服务器端输出上传结果.必须.不然客户端接受不到指定responseJason参数用来判断上传后状态.
在进一步看看服务器端如何处理上传的找到handleUpload函数定义.

/** 
* Handle the uploaded file 
* @param string $uploadDirectory 
* @param string $replaceOldFile=true 
* @returns array('success'=>true) or array('error'=>'error message') 
*/ 
function handleUpload($uploadDirectory, $replaceOldFile = FALSE){ 
if (!is_writable($uploadDirectory)){ 
return array('error' => "Server error. Upload directory isn't writable."); 
} if (!$this->file){ 
return array('error' => 'No files were uploaded.'); 
} 
$size = $this->file->getSize(); 
if ($size == 0) { 
return array('error' => 'File is empty'); 
} 
if ($size > $this->sizeLimit) { 
return array('error' => 'File is too large'); 
} 
$pathinfo = pathinfo($this->file->getName()); 
$filename = $pathinfo['filename']; 
//$filename = md5(uniqid()); 
$ext = @$pathinfo['extension'];// hide notices if extension is empty 
if($this->allowedExtensions && !in_array(strtolower($ext), $this->allowedExtensions)){ 
$these = implode(', ', $this->allowedExtensions); 
return array('error' => 'File has an invalid extension, it should be one of '. $these . '.'); 
} 
$ext = ($ext == '') ? $ext : '.' . $ext; 
if(!$replaceOldFile){ 
/// don't overwrite previous files that were uploaded 
while (file_exists($uploadDirectory . DIRECTORY_SEPARATOR . $filename . $ext)) { 
$filename .= rand(10, 99); 
} 
} 
$this->uploadName = $filename . $ext; 
if ($this->file->save($uploadDirectory . DIRECTORY_SEPARATOR . $filename . $ext)){ 
return array('success'=>true); 
} else { 
return array('error'=> 'Could not save uploaded file.' . 
'The upload was cancelled, or server error encountered'); 
} 
}

在调用这个处理函数时.需要注意的是.传递的URL存储路径需要时绝对的.所以需要对传入路劲做一下格式化处理:
$uploadDirectory = $_SERVER['DOCUMENT_ROOT']."DS".$uploadDirectory;
对于is_writeable文件是否可写的判断.我个人认为还不够详细.is_writeable主要判断文件或目录是否存在.并可写才会返回true. 所以个人建议在is_writable前添加一个文件是否存在.这样更易于在客户端判断服务器端文件出错具体的情况:
if (!file_exists($uploadDirectory)) { 
return array('error' => "Server error. Upload directory dones't exist."); 
}

在保存文件操作前.可以看到.处理函数分别做了四次判断.分别判断了 上传文件的数量、文件上传的大小、文件上传大小是否超过上限、另外在上传过程.如果我们多次想服务器端上传同一个文件.发现Fine Uploder处理方式是.并非是重写.而是从10-99随机一个数字重写命名该文件.并保存到目录下.当保存文件成功后.则想服务器端返回一个Json数据其中包含IsSuccess来指定此次上传操作是否操作成功. IsSuccess参数作为客户端判断此时操作唯一参数.

在上传操作过程发信很多出现“increase post_max_size and upload_max_filesize to 10M”错误,其实针对这个问题.主要是上传文件配置超过php环境默认的2M.需要在php.ini文件中把post_max_size和upload_max_filesize两项的值改到10M以上,然后重启Apache即可.或是参考Php官网针对配置说明 修改php.ini配置文件.
至此整个Fine Uploader配置流程已经全部完成.点击选择文件时.会如下效果:
Fine Uploader文件上传组件应用介绍 
提示上传成功.当然更多的请参考官方给出演示demo.如上从Fine Uploader源码角度分析其实现原理.

PHP 相关文章推荐
PHP个人网站架设连环讲(二)
Oct 09 PHP
php获取目录所有文件并将结果保存到数组(实例)
Oct 25 PHP
php模拟登陆的实现方法分析
Jan 09 PHP
php递归法读取目录及文件的方法
Jan 30 PHP
PHP获取QQ达人QQ信息的方法
Mar 05 PHP
php相对当前文件include其它文件的方法
Mar 13 PHP
Thinkphp无限级分类代码
Nov 11 PHP
php用户密码加密算法分析【Discuz加密算法】
Oct 12 PHP
PHP自动补全表单的两种方法
Mar 06 PHP
Yii2.0 RESTful API 基础配置教程详解
Dec 26 PHP
PHP设计模式之工厂模式(Factory Pattern)的讲解
Mar 21 PHP
Laravel使用swoole实现websocket主动消息推送的方法介绍
Oct 20 PHP
PDO版本问题 Invalid parameter number: no parameters were bound
Jan 06 #PHP
PHPEXCEL 使用小记
Jan 06 #PHP
关于mysql字符集设置了character_set_client=binary 在gbk情况下会出现表描述是乱码的情况
Jan 06 #PHP
Could not load type System.ServiceModel.Activation.HttpModule解决办法
Dec 29 #PHP
PHP数组无限分级数据的层级化处理代码
Dec 29 #PHP
php学习笔记之面向对象编程
Dec 29 #PHP
php5.3中连接sqlserver2000的两种方法(com与ODBC)
Dec 29 #PHP
You might like
ie6 动态缩略图不显示的原因
2009/06/21 PHP
php getsiteurl()函数
2009/09/05 PHP
PHP采集类Snoopy抓取图片实例
2014/06/19 PHP
linux中cd命令使用详解
2015/01/08 PHP
微信随机生成红包金额算法php版
2016/07/21 PHP
tp5(thinkPHP5)框架连接数据库的方法示例
2018/12/24 PHP
PHP实现的操作数组类库定义与用法示例
2019/05/24 PHP
PHP xpath提取网页数据内容代码解析
2020/07/16 PHP
PHP中SESSION过期设置
2021/03/09 PHP
读jQuery之二(两种扩展)
2011/06/11 Javascript
用js实现table单元格高宽调整,兼容合并单元格(兼容IE6、7、8、FF)实例
2013/06/25 Javascript
Javascript解析URL方法详解
2014/12/05 Javascript
javascript实现数独解法
2015/03/14 Javascript
jQuery+css3实现Ajax点击后动态删除功能的方法
2015/08/10 Javascript
基于JavaScript代码实现自动生成表格
2016/06/15 Javascript
js转html实体的方法
2016/09/27 Javascript
简单实现jQuery弹幕效果
2017/05/06 jQuery
微信小程序本地缓存数据增删改查实例详解
2017/05/24 Javascript
vue实现图书管理demo详解
2017/10/17 Javascript
详解利用eventemitter2实现Vue组件通信
2019/11/04 Javascript
解决nuxt页面中mounted、created、watch执行两遍的问题
2020/11/05 Javascript
解决Python传递中文参数的问题
2015/08/04 Python
pandas数据处理之绘图的实现
2020/06/15 Python
python3实现语音转文字(语音识别)和文字转语音(语音合成)
2020/10/14 Python
H5仿微信界面教程(一)
2017/07/05 HTML / CSS
美国独家设计师眼镜在线光学商店:Glasses Gallery
2017/12/28 全球购物
prAna官网:瑜伽、旅行和冒险服装
2019/03/10 全球购物
Guess欧洲官网:美国服饰品牌
2019/08/06 全球购物
销售类个人求职信范文
2013/09/25 职场文书
工程建设实施方案
2014/03/14 职场文书
2014年度安全生产目标管理责任书
2014/07/25 职场文书
公务员年度个人总结
2015/02/12 职场文书
2015年宣传思想工作总结
2015/05/22 职场文书
2016年感恩教师节校园广播稿
2015/12/18 职场文书
2016大学生诚信考试承诺书
2016/03/25 职场文书
go语言中fallthrough的用法说明
2021/05/06 Golang