Angular下H5上传图片的方法(可多张上传)
最近做的项目中用到了angular下上传图片功能,在做的过程中遇到了许多问题,最终都得以解决
angular上传时和普通上传时过程差不多,只不过是要不一些东西转化为angular的东西。
1.ng-file-select,指令angular是没此功能的,其实也是转化成了change事件,不多说,直接上代码
angular.module('myapp') .directive('ngfileselect', [ '$parse', '$timeout', function($parse, $timeout) { return function(scope, elem, attr) { var fn = $parse(attr['ngfileselect']); elem.bind('change', function(evt) { var files = [], 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) { var datauritoblob = function(datauri) { // convert base64/urlencoded data component to raw binary data held in a string var bytestring; if (datauri.split(',')[0].indexof('base64') >= 0) bytestring = atob(datauri.split(',')[1]); else bytestring = unescape(datauri.split(',')[1]); // separate out the mime component var mimestring = datauri.split(',')[0].split(':')[1].split(';')[0]; // write the bytes of the string to a typed array var ia = new uint8array(bytestring.length); for (var i = 0; i < bytestring.length; i++) { ia[i] = bytestring.charcodeat(i); } return new blob([ia], { type: mimestring }); }; var onload = function(reader, deferred, scope,file) { return function() { scope.$apply(function() { var img = new image(); //前端压缩图片 img.onload = function(){ //resize the image using canvas var canvas = document.createelement("canvas"); var ctx = canvas.getcontext("2d"); var width = img.width; var height = img.height; var max_width = width>2500 ? width/2 : 2500; var max_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); var dataurl = canvas.todataurl('image/jpeg', 1); var blob = datauritoblob(dataurl); if(blob.size > 2000 * 1024){ dataurl = canvas.todataurl('image/jpeg', .2); }else if(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); }); }; }; var onerror = function(reader, deferred, scope) { return function() { scope.$apply(function() { deferred.reject(reader.result); }); }; }; var onprogress = function(reader, scope) { return function(event) { scope.$broadcast("fileprogress", { total: event.total, loaded: event.loaded }); }; }; var getreader = function(deferred, scope, file) { var reader = new filereader(); reader.onload = onload(reader, deferred, scope,file); reader.onerror = onerror(reader, deferred, scope); reader.onprogress = onprogress(reader, scope); return reader; }; var readasdataurl = function(file, scope) { var deferred = $q.defer(); var reader = getreader(deferred, scope,file); reader.readasdataurl(file); return deferred.promise; }; return { readasdataurl: readasdataurl }; }]);
这里说明一下,部分代码是参考别人的代码(),但是对其中内容做了修改,因为用原来的代码,如果不加前端压缩功能是正常的,前端压缩的话因为要用到canvas, 直接用reader.result在ios上图片的宽高拿到的直接是0,android上是可以的,具体原因不是很确定是不是base64的问题,所以我又直接把file传了进来,然后用原生js的方法新建图片元素拿到宽高,再用canvas进行压缩,最后转成blob,通过formdata传给后台。
3.controller代码
//选择图片后执行的方法 $scope.filearr = []; $scope.imgsrcarr = [];var i = 0; //为ios上图片都为image时添加序号 $rootscope.onfileselect = function(files, event) { //预览上传图片开始 $rootscope.startloading(); var $this = angular.element(event.target); angular.foreach(files, function(value, index) { var filein = value; var fileinname = filein.name; var filetype = 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; };return false; }
这里处理了下图片,在名字上加了序号,因为在ios上每次选择的图片名字都叫image,查找了很多资料,说是safari的bug,后面版本才会解决,暂时只能以这种方式解决了。循环是上传多张图片
3.html代码
<ul class="upload-view-ul"> <li ng-repeat="src in imgsrcarr" class="pull-left" ng-click="delcurupload(src)" ng-class="{'row-last': (($index+1) % 5==0)}"> <span>x</span> <em ng-if='nrc'>{{formdata.attachvo[$index].attachmenttype}}</em> <img ng-src="{{src}}"> </li> <div class="attachment" pop-type-select ng-if="nrc">+</div> <div class="attachment" ng-if="!nrc"> + <form class="upload"> <input type="file" name="file[]" ng-file-select="onfileselect($files, $event)" multiple> </form> </div> </ul>
4.顺便把formdata时代码贴一下,采用h5上传图片的方式
this.formdatapost = function(pathurl, formid, formdata, files) { var fd = new formdata(); 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)); var httpconfig = { headers: { 'authorization': 'bearer ' + this.token, 'content-type': undefined }, transformrequest: angular.identity }; return $http.post(rooturl + pathurl, fd, httpconfig).then(function(data) { return data; }).catch(function(error) { $rootscope.interfacename = pathurl; $rootscope.setnewwortstatus({ status: error.status, errinfo: error.data && error.data.statusinfo || '' }); return error; }); }
思路有一点混乱,不知道讲清楚了没有,想起来再添加吧
以上所述是小编给大家介绍的angular下h5上传图片的方法(可多张上传),希望对大家有所帮助
上一篇: cdr中怎么制作三维效果的线条?
下一篇: 步步高集团 开启大数据时代新征程
推荐阅读