上传文件 file upload 学习笔记
这里我只会说说一些完成 file upload 的基础 API。
很多项目我们需要上传文件。
有简单的 input file, 有需要验证的,有需要压缩的(img),有需要分段的(video),有需要体验好(display on local, ajax & percent) 等等
要完成以上的所有需求,我们需要很多底层的 API, 比如 File, FileReader, Canvas , XMLHttpRequest , Blob
要把这个控件写好的话,需要一些设计模式,不过这篇不会涉及这个,我只是想大略的说说过程和使用到的API方法,如何去设计就交给你们自己了。
参考 :
http://javascript.ruanyifeng.com/bom/ajax.html
http://www.html5rocks.com/zh/tutorials/file/dndfiles/
https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toDataURL
http://www.w3schools.com/tags/canvas_drawimage.asp
https://developer.mozilla.org/en-US/docs/Web/API/Blob/slice
http://www.codicode.com/art/upload_and_save_a_canvas_image_to_the_server.aspx
基本的步骤是这样的
1. input file addEventListener change 监听 input file (你要用 drag drop 替代也行)
2. 得到 file 对象之后你就可以检查 file.name,file.type,file.size 等等了
3. 我可以通过 FileReader.readAsDataURL(file); 来获取一个 base64 string (这个是针对图片的处理,如果是txt的话你可以用别的 FileReader.readAs...其它)
4. 把这个 base64 放入img.src = base64 的话就可以在本地显示图片了
5. 在打开一个canvas 后,我们可以把上面的这张相片放入 canvas , 修改它的尺寸,加水印等等.
6. canvas.toDataURL() 把图像转换回 base64 string.
7. 通过XMLHttpRequest 上传这个 base64 string. (XMLHttpRequest 可以监听 process percent)
8. 后端把 base64 convert to byte 写入 fileStream 就完成了。
如果要支持分段上传的话,base64应该直接substring记入index position 就可以了。
如果上传的是 file 不是 base64 的话,可以使用 Blob.webkitSlice() | Blob.slice
document.getElementById("file").addEventListener("change", function (e) {
var files = S.toArray(this.files);
var file = files[0];
var fileReader = new FileReader(); fileReader.onloadend = function (e) {
var img = new Image();
img.onload = function () {
//比例要是尺寸
var MAX_WIDTH = 2560 / 2;
var canvas = document.getElementById("canvas");
var context = canvas.getContext("2d");
var height = this.height * (MAX_WIDTH / this.width);
canvas.width = MAX_WIDTH;
canvas.height = height;
//画图
//参数就是把img 的x,y,widthLength,heightLength copy
//然后 paste to canvas 的 x,y,widthLength,heightLength
context.drawImage(this, 0, 0, this.width, this.height, 0, 0, MAX_WIDTH, height); var base64 = canvas.toDataURL("image/jpeg", 0.8); //default 会是 png 格式,第2参数是压缩质量 0-1 (png 不可以压缩)
base64 = base64.replace(/^data:image\/(png|jpeg);base64,/, ""); //后端保存的时候不可以有前面这些字,所以在这里先去除 var http = new XMLHttpRequest();
var formData = new FormData();
formData.append("file", base64);
http.open("POST", "//localhost:8054/module/Upload/UploadAjax.ashx", true);
http.onreadystatechange = function () {
if (this.readyState == 4) {
if (this.status == 200) {
var imgSrc = this.responseText;
}
else if (this.status == 0) { //abort 会"同步"执行这里 }
else {
log("file upload ajax fail, looping upload stoped. statusCode : " + this.status);
}
}
}
http.upload.onprogress = function (e) {
if (e.lengthComputable) {
var percent = parseFloat(e.loaded / e.total * 100).toFixed(0); //八仙
}
}
http.send(formData);
}
img.src = this.result;
}
fileReader.readAsDataURL(file); }, false);
//upload base64 的处理
string path = context.Server.MapPath(@"~\img\" + "abc.jpeg");
using (FileStream fs = new FileStream(path, FileMode.Create))
{
using (BinaryWriter bw = new BinaryWriter(fs))
{
byte[] data = Convert.FromBase64String(context.Request.Form["file"]);
bw.Write(data);
bw.Close();
}
}
context.Response.ContentType = "text/plain";
context.Response.Write(path);
//update file 的处理
string[] fileKeys = context.Request.Files.AllKeys;
string fileName = "";
foreach (string fileKey in fileKeys)
{
HttpPostedFile file = context.Request.Files[fileKey];
string extension = "." + file.ContentType.Substring();
//string extension = file.FileName.Substring(file.FileName.LastIndexOf("."));
fileName = Guid.NewGuid() + extension;
file.SaveAs(context.Server.MapPath(@"~\img\" + fileName));
}
context.Response.ContentType = "text/plain";
context.Response.Write(fileName);
有一点要说明一下,canvas to base64 png 的话,size 是非常大的, 比源文件还要大的多,所以png 格式是不能通过canvas做压缩的. (如果我错了,请告诉我)
所以一般上我们是对jpeg做压缩, 质量设置成0.92 的话,size 会和源文件相同. 不过0.92并不是default的设置,default设置会再低一些。
上面我没有实现水印和分段,以后等我有时间写一个完整的控件时,我才更新。
上传文件 file upload 学习笔记的更多相关文章
- DVWA 黑客攻防演练(五)文件上传漏洞 File Upload
说起文件上传漏洞 ,可谓是印象深刻.有次公司的网站突然访问不到了,同事去服务器看了一下.所有 webroot 文件夹下的所有文件都被重命名成其他文件,比如 jsp 文件变成 jsp.s ,以致于路径映 ...
- vuetify | vue | 文件上传组件 | file | upload | form input[type="file"]
今天无聊地写vuecli3听歌的时候,遇到了上传文件到Django的自我需求,然后就到vuetify的表单组件里找upload btn,发现居然没有!!! 顿时惊了个呆,要知道之前用element做操 ...
- Ajax JQuery HTML 提交上传文件File HTML+ Ajax+ASP.NET+ WebService
起因:公司最近有些项目用到了HTML+WebService的组合,发现访问速度等都快了许多,但是由于通过Ajax只能应付一些简单的文字类的传输,上传文件就捉襟见肘了,如果一直引用第三方的swf之类上传 ...
- php Socket模拟表单上传文件函数_学习
模拟上传文件的php代码 里面访问地址.主机.上传文件名.内容.分隔符可以修改 function postFile($file) { $clf = "\r\n"; ...
- Spring MVC-表单(Form)标签-文件上传(File Upload)示例(转载实践)
以下内容翻译自:https://www.tutorialspoint.com/springmvc/springmvc_upload.htm 说明:示例基于Spring MVC 4.1.6. 以下示例显 ...
- JQuery上传文件插件Uploadify使用笔记
新工作的第一份任务就是给实现 限制Uploadify 上传文件格式为图片 测试出来报错,选择了非图片文件,提示错误后,再选择其他文件,上传时还是包含了之前清空的非图片文件 最后实现效果的代码是 //上 ...
- 使用vue+iview实现上传文件及常用的下载文件的方法
首先说明一下,我们这次主要用的还是iview的upload上传组件,下面直接上代码 <Upload ref="upload" multiple='true' //是否支持多文 ...
- SpringMVC+jquery.uploadify 上传文件
前言 以前用Asp.net MVC+uploadify上传文件,最近学习SpringMVC,所以就用SpringMVC+uploadify做个上传文件的demo. 刚开始用form表单的方式提交,在C ...
- JAVA模拟HTTP post请求上传文件
在开发中,我们使用的比较多的HTTP请求方式基本上就是GET.POST.其中GET用于从服务器获取数据,POST主要用于向服务器提交一些表单数据,例如文件上传等.而我们在使用HTTP请求时中遇到的比较 ...
随机推荐
- F - Count the Colors - zoj 1610(区间覆盖)
有一块很长的画布,现在想在这块画布上画一些颜色,不过后面画的颜色会把前面画的颜色覆盖掉,现在想知道画完后这块画布的颜色分布,比如 1号颜色有几块,2号颜色有几块.... *************** ...
- 安装VS2012 update3提示缺少Microsoft根证书颁发机构2010或2011的解决方法
警告提示如图: (copy的百度贴吧的童鞋的截图) 解决方法: 下载2010.10或2011.10的根证书即可 直通车:http://maxsky.ys168.com/ ——05.||浮云文件||—— ...
- SKPhysicsJoint类
继承自 NSObject 符合 NSCodingNSObject(NSObject) 框架 /System/Library/Frameworks/SpriteKit.framework 可用性 可用 ...
- 详述iOS国际化
在真正将国际化实践前,只知道通过NSLocalizedString方法将相应语言的字符串加载进来即可.但最近公司项目的新需求增加英文版本,并支持应用内无死角切换~,这才跳过各种坑实现了应用内切换语言, ...
- Java 编程的动态性,第 6 部分: 利用 Javassist 进行面向方面的更改--转载
本系列的 第 4 部分和 第 5 部分讨论了如何用 Javassist 对二进制类进行局部更改.这次您将学习以一种更强大的方式使用该框架,从而充分利用 Javassist 对在字节码中查找所有特定方法 ...
- MediaPlayer+SurfaceView 视频播放 示例
SurfaceView的原理 SurfaceView在视频播放中起到显示画面的作用,而视频的播放主要通过MediaPlayer来控制. SurfaceView 允许我们 ...
- maven第5章坐标和依赖
5.5依赖范围 runtime:运行时依赖范围 举的例子是JDBC驱动实现,不理解? 编译的时候只需要引入jdk提供的jdbc类和方法,比如java.sql.*;这时编译没有报错,但是没有引入mysq ...
- tomcat startup.sh提示java.lang.OutOfMemoryError: PermGen space
JAVA_OPTS="-server -XX:PermSize=512M -XX:MaxPermSize=1024m"if [ -z "$LOGGING_MANAGER& ...
- 把数据库中的null作为条件查询应该用is
如select * from mbXX where tuijian is null 而不是select * from mbXX where tuijian=null
- 连接、关联、JOIN、APPLY(SQL Server)
连接方式 连接类型 个人总结 阐述(生成两个集合的约束笛卡儿积) INNER JOIN 内连接 关联相同的(用于查找关联的信息) FROM C AS c INNER JOIN D AS d ON ...