Angular下H5上传图片的方法(可多张上传)
最近做的项目中用到了angular下上传图片功能,在做的过程中遇到了许多问题,最终都得以解决
angular上传时和普通上传时过程差不多,只不过是要不一些东西转化为angular的东西。
1.ng-file-select,指令angular是没此功能的,其实也是转化成了change事件,不多说,直接上代码
angular.module('myApp')
.directive('ngFileSelect',['$parse','$timeout',function($parse,$timeout){
returnfunction(scope,elem,attr){
varfn=$parse(attr['ngFileSelect']);
elem.bind('change',function(evt){
varfiles=[],fileList,i;
fileList=evt.target.files;
if(fileList!=null){
for(i=0;i<fileList.length;i++){
files.push(fileList.item(i));
}
}
$timeout(function(){
fn(scope,{
$files:files,
$event:evt
});
});
});
};
}])
2.服务上传文件前预览并压缩图片功能
//上传文件预览
angular.module('myServers',[])
.factory('fileReader',['$q','$log',function($q,$log){
vardataURItoBlob=function(dataURI){
//convertbase64/URLEncodeddatacomponenttorawbinarydataheldinastring
varbyteString;
if(dataURI.split(',')[0].indexOf('base64')>=0)
byteString=atob(dataURI.split(',')[1]);
else
byteString=unescape(dataURI.split(',')[1]);
//separateoutthemimecomponent
varmimeString=dataURI.split(',')[0].split(':')[1].split(';')[0];
//writethebytesofthestringtoatypedarray
varia=newUint8Array(byteString.length);
for(vari=0;i<byteString.length;i++){
ia[i]=byteString.charCodeAt(i);
}
returnnewBlob([ia],{
type:mimeString
});
};
varonLoad=function(reader,deferred,scope,file){
returnfunction(){
scope.$apply(function(){
varimg=newImage();
//前端压缩图片
img.onload=function(){
//resizetheimageusingcanvas
varcanvas=document.createElement("canvas");
varctx=canvas.getContext("2d");
varwidth=img.width;
varheight=img.height;
varMAX_WIDTH=width>2500?width/2:2500;
varMAX_HEIGHT=height>2500?height/2:2500;
if(width>height){
if(width>MAX_WIDTH){
height*=MAX_WIDTH/width;
width=MAX_WIDTH;
}
}else{
if(height>MAX_HEIGHT){
width*=MAX_HEIGHT/height;
height=MAX_HEIGHT;
}
}
canvas.width=width;
canvas.height=height;
ctx.drawImage(img,0,0,width,height);
vardataURL=canvas.toDataURL('image/jpeg',1);
varblob=dataURItoBlob(dataURL);
if(blob.size>2000*1024){
dataURL=canvas.toDataURL('image/jpeg',.2);
}elseif(blob.size>1000*1024){
dataURL=canvas.toDataURL('image/jpeg',.5);
}else{
dataURL=canvas.toDataURL('image/jpeg',.8);
}
blob=dataURItoBlob(dataURL);
deferred.resolve(blob);
}
img.src=URL.createObjectURL(file);
});
};
};
varonError=function(reader,deferred,scope){
returnfunction(){
scope.$apply(function(){
deferred.reject(reader.result);
});
};
};
varonProgress=function(reader,scope){
returnfunction(event){
scope.$broadcast("fileProgress",{
total:event.total,
loaded:event.loaded
});
};
};
vargetReader=function(deferred,scope,file){
varreader=newFileReader();
reader.onload=onLoad(reader,deferred,scope,file);
reader.onerror=onError(reader,deferred,scope);
reader.onprogress=onProgress(reader,scope);
returnreader;
};
varreadAsDataURL=function(file,scope){
vardeferred=$q.defer();
varreader=getReader(deferred,scope,file);
reader.readAsDataURL(file);
returndeferred.promise;
};
return{
readAsDataUrl:readAsDataURL
};
}]);
这里说明一下,部分代码是参考别人的代码(http://blog.csdn.net/zx007fack/article/details/41073601),但是对其中内容做了修改,因为用原来的代码,如果不加前端压缩功能是正常的,前端压缩的话因为要用到canvas,直接用reader.result在ios上图片的宽高拿到的直接是0,android上是可以的,具体原因不是很确定是不是base64的问题,所以我又直接把file传了进来,然后用原生js的方法新建图片元素拿到宽高,再用Canvas进行压缩,最后转成blob,通过formData传给后台。
3.controller代码
//选择图片后执行的方法
$scope.fileArr=[];
$scope.imgSrcArr=[];vari=0;//为ios上图片都为image时添加序号
$rootScope.onFileSelect=function(files,event){
//预览上传图片开始
$rootScope.startLoading();
var$this=angular.element(event.target);
angular.forEach(files,function(value,index){
varfileIn=value;
varfileInName=fileIn.name;
varfileType=fileInName.substring(fileInName.lastIndexOf(".")+1,fileInName.length);
//解决ios下所有图片都为image.jpg的bug
if(fileIn){
fileInName=fileInName.split('.')[0]+i+'.'+fileType;
i++;
}
attachvo.push({
name:fileInName,
type:fileType
});
fileReader.readAsDataUrl(fileIn,$scope)
.then(function(result){
result.name=fileInName;
$scope.fileArr.push(result);
$scope.imgSrcArr.push(URL.createObjectURL(result));
//每次上传后清空file框,确保每次都能调用change事件
document.querySelector('.upload').reset();
});
$scope.$on('fileProgress',function(event,data){
if(data.total==data.loaded){
$timeout(function(){
//上传图片结束
$rootScope.endLoading();
},200)
}
});
});
$rootScope.showAttachment=false;
};returnfalse;
}
这里处理了下图片,在名字上加了序号,因为在ios上每次选择的图片名字都叫image,查找了很多资料,说是safari的bug,后面版本才会解决,暂时只能以这种方式解决了。循环是上传多张图片
3.html代码
<ulclass="upload-view-ul">
<ling-repeat="srcinimgSrcArr"class="pull-left"ng-click="delCurUpload(src)"
ng-class="{'row-last':(($index+1)%5==0)}">
<span>x</span>
<emng-if='nrc'>{{formData.attachvo[$index].attachmentType}}</em>
<imgng-src="{{src}}">
</li>
<divclass="attachment"pop-type-selectng-if="nrc">+</div>
<divclass="attachment"ng-if="!nrc">
+
<formclass="upload">
<inputtype="file"name="file[]"ng-file-select="onFileSelect($files,$event)"multiple>
</form>
</div>
</ul>
4.顺便把formdata时代码贴一下,采用H5上传图片的方式
this.FormdataPost=function(pathUrl,formId,formData,files){
varfd=newFormData();
fd.append('formId',formId);
if(files&&angular.isArray(files)){
files.forEach(function(item){
fd.append('file',item,item.name);
});
}
fd.append('formData',angular.toJson(formData,true));
varhttpConfig={
headers:{
'Authorization':'Bearer'+this.token,
'Content-Type':undefined
},
transformRequest:angular.identity
};
return$http.post(rootUrl+pathUrl,fd,httpConfig).then(function(data){
returndata;
}).catch(function(error){
$rootScope.interfaceName=pathUrl;
$rootScope.setNewWortStatus({
status:error.status,
errInfo:error.data&&error.data.statusInfo||''
});
returnerror;
});
}
思路有一点混乱,不知道讲清楚了没有,想起来再添加吧
以上所述是小编给大家介绍的Angular下H5上传图片的方法(可多张上传),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!