Angular下H5上傳圖片的方法(可多張上傳)
最近做的項(xiàng)目中用到了angular下上傳圖片功能,在做的過程中遇到了許多問題,最終都得以解決
angular上傳時(shí)和普通上傳時(shí)過程差不多,只不過是要不一些東西轉(zhuǎn)化為angular的東西。
1.ng-file-select,指令angular是沒此功能的,其實(shí)也是轉(zhuǎn)化成了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.服務(wù) 上傳文件前預(yù)覽并壓縮圖片功能
//上傳文件預(yù)覽
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
};
}]);
這里說明一下,部分代碼是參考別人的代碼(http://blog.csdn.net/zx007fack/article/details/41073601),但是對其中內(nèi)容做了修改,因?yàn)橛迷瓉淼拇a,如果不加前端壓縮功能是正常的,前端壓縮的話因?yàn)橐玫絚anvas, 直接用reader.result在ios上圖片的寬高拿到的直接是0,android上是可以的,具體原因不是很確定是不是base64的問題,所以我又直接把file傳了進(jìn)來,然后用原生js的方法新建圖片元素拿到寬高,再用Canvas進(jìn)行壓縮,最后轉(zhuǎn)成blob,通過formData傳給后臺。
3.controller代碼
//選擇圖片后執(zhí)行的方法
$scope.fileArr = [];
$scope.imgSrcArr = [];var i = 0; //為ios上圖片都為image時(shí)添加序號
$rootScope.onFileSelect = function(files, event) {
//預(yù)覽上傳圖片開始
$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框,確保每次都能調(diào)用change事件
document.querySelector('.upload').reset();
});
$scope.$on('fileProgress', function(event, data) {
if(data.total == data.loaded) {
$timeout(function() {
//上傳圖片結(jié)束
$rootScope.endLoading();
}, 200)
}
});
});
$rootScope.showAttachment = false;
};return false;
}
這里處理了下圖片,在名字上加了序號,因?yàn)樵趇os上每次選擇的圖片名字都叫image,查找了很多資料,說是safari的bug,后面版本才會解決,暫時(shí)只能以這種方式解決了。循環(huán)是上傳多張圖片
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時(shí)代碼貼一下,采用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;
});
}
思路有一點(diǎn)混亂,不知道講清楚了沒有,想起來再添加吧
以上所述是小編給大家介紹的Angular下H5上傳圖片的方法(可多張上傳),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
- angularjs客戶端實(shí)現(xiàn)壓縮圖片文件并上傳實(shí)例
- 通過AngularJS實(shí)現(xiàn)圖片上傳及縮略圖展示示例
- AngularJs上傳前預(yù)覽圖片的實(shí)例代碼
- angularjs實(shí)現(xiàn)多張圖片上傳并預(yù)覽功能
- angularjs點(diǎn)擊圖片放大實(shí)現(xiàn)上傳圖片預(yù)覽
- angular2+nodejs實(shí)現(xiàn)圖片上傳功能
- Angularjs上傳圖片實(shí)例詳解
- Angularjs實(shí)現(xiàn)上傳圖片預(yù)覽功能
- AngularJS實(shí)現(xiàn)圖片上傳和預(yù)覽功能的方法分析
- Angular4實(shí)現(xiàn)圖片上傳預(yù)覽路徑不安全的問題解決
相關(guān)文章
對angular 監(jiān)控?cái)?shù)據(jù)模型變化的事件方法$watch詳解
今天小編就為大家分享一篇對angular 監(jiān)控?cái)?shù)據(jù)模型變化的事件方法$watch詳解,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-10-10
使用AngularJS實(shí)現(xiàn)可伸縮的頁面切換的方法
這篇文章主要介紹了使用AngularJS實(shí)現(xiàn)可伸縮的頁面切換的方法,AngularJS是一款熱門的JavaScript庫,需要的朋友可以參考下2015-06-06
Angular JS 生成動(dòng)態(tài)二維碼的方法
這篇文章主要介紹了Angular JS 生成動(dòng)態(tài)二維碼的方法,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-02-02
AngularJS基礎(chǔ) ng-focus 指令簡單示例
本文主要介紹AngularJS ng-focus 指令,這里整理了ng-focus的一些基礎(chǔ)資料,并附一個(gè)實(shí)例代碼,有需要的小伙伴參考下2016-08-08
Angular中Lazy Loading懶加載陷阱避坑最佳實(shí)踐
這篇文章主要為大家介紹了Angular中Lazy Loading懶加載陷阱避坑最佳實(shí)踐,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10
angular6 利用 ngContentOutlet 實(shí)現(xiàn)組件位置交換(重排)
這篇文章主要介紹了angular6 利用 ngContentOutlet 實(shí)現(xiàn)組件位置交換(重排),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-11-11
AngularJS ng-bind 指令簡單實(shí)現(xiàn)
本文主要介紹AngularJS ng-bind 指令,在這里對ng-bind 指令做了詳細(xì)資料整理并講解,提供了實(shí)例代碼以便大家參考,有需要的小伙伴可以參考下2016-07-07

