原文出处:http://shazwazza.com/post/uploading-files-and-json-data-in-the-same-request-with-angular-js/
我决定写一遍快速介绍这方面的文章,因为很多关于这方面的文档或例子都太片面了。通过这篇文章,我们将学会实现用同样的请求处理来上传任意数量的文件和其他任意格式的数据。这个例子我们会上传一些Json格式的数据和文件。
app.directive('fileUpload',function () {然后,很简单的应用它:
return {
scope: true,128,0)">//create a new scope
link: function (scope,el,attrs) {
el.bind('change',255)">function (event) {
var files = event.target.files;
//iterate files since 'multiple' may be specified on the element
for (var i = 0;i<files.length;i++) {
//emit event upward
scope.$emit("fileSelected",{ file: files[i] });
}
});
}
};
});
<input type="file" file-upload multiple/>multiple属性说明这个例子支持用户可以选择多个文件进行上传。
在上面指令中,我们要确定建立一个新的作用域。然后,去监听 input file元素的change事件。当监测到文件对象变化时,将会触发作用域中的事件,以及其父作用域中的事件(向上层)。
创建控制器
接下来,我们要创建一个控制器:
1.创建一个绑定它的模型
2.创建一个文件集合
3.创建一个绑定事件的处理程序,以便我们将文件放入文件集合中。
备注:为了更快捷,我将所有的方法都放到一个控制器中。在大多数情况下,你需要将它们放到不同的处理数据的工厂方法中。
控制器放在这里,这个标注用起来像下面这样(将会显示所有选择的文件)
div ng-controller="Ctrl">在控制器中,如下面的代码含有一些很重要的关于如何获得上传到服务器端数据的备注,协议头中 Content-Type 一定要被设置,看起来很诡异吧。
/>
ul>
li ng-repeat="file in files">{{file.name}}</li>
div>
//译者备注:这段很重要,重点之关键
function Ctrl($scope,$http) {
//a simple model to bind to and send to the server
$scope.model = {
name: "",
comments: ""
};
//an array of files selected
$scope.files = [];
//listen for the file selected event
$scope.$on("fileSelected" (event$scope.$apply( () {
//add the file object to the scope's files collection
$scope.files.push(args.file);
});
//the save method
$scope.save = () {
$http({
method: 'POST'url: "/Api/PostStuff"//IMPORTANT!!! You might think this should be set to 'multipart/form-data'
// but this is not true because when we are sending up files the request
// needs to include a 'boundary' parameter which identifies the boundary
// name between parts in this multi-part request and setting the Content-type
// manually will not set this boundary parameter. For whatever reason,0)">// setting the Content-type to 'false' will force the request to automatically
// populate the headers properly including the boundary parameter.
headers: { 'Content-Type': false },0)">//This method will allow us to change how the data is sent up to the server
// for which we'll need to encapsulate the model data in 'FormData'
transformRequest: (data) {
var formData = new FormData();
//need to convert our json object to a string version of json otherwise
// the browser will do a 'toString()' on the object which will result
// in the value '[Object object]' on the server.
formData.append("model"//now add all of the assigned files
for i = 0; i < data.files; i++) {
//add each file to the form data and iteratively name them
"file" + i,data.files[i]);
}
return formData;
},0)">//Create an object that contains the model and files which will be transformed
// in the above transformRequest method
data: { model: $scope.model,files: $scope.files }
}).
success( (data,status,headers,config) {
alert("success!");
error("Failed!"};@H_283_301@
处理服务器端的数据
public async Task<HttpResponseMessage> PostStuff()
{
if (!Request.Content.IsMimeMultipartContent())
{
throw new HttpResponseException(HttpStatusCode.UnsupportedMediaType);
}
var root = HttpContext.Current.Server.MapPath("~/App_Data/Temp/FileUploads");
Directory.CreateDirectory(root);
var provider = new MultipartFormDataStreamProvider(root);
var result = await Request.Content.ReadAsMultipartAsync(provider);
if (result.FormData["model"] == null)
{
new HttpResponseException(HttpStatusCode.BadRequest);
}
var model = result.FormData["model"];
//TODO: Do something with the json model which is currently a string
//get the files
foreach (var file in result.FileData)
{
//TODO: Do something with each uploaded file
}
return Request.CreateResponse(HttpStatusCode.OK,128)">"success!");}