如何通过HttpPostedFileBase和ViewModel一起发送上传的文件?

时间:2016-04-15 09:21:42

标签: c# jquery ajax asp.net-mvc

我有一个巨大的数据对象,应该通过控制器参数映射到视图模型。我还有一个文件列表(照片),我需要在上述数据对象的数组中捕获。

我的模特:

public class TestPerson
{
    public int Id { get; set; }
    public string Name { get; set; }
    public string Place { get; set; }
    public int Age { get; set; }
    public byte[] File { get; set; }

    //[FileSize(10240)]
    //[FileTypes("jpg,jpeg,png")]
    //public HttpPostedFileBase File { get; set; }
}

我的控制器方法:

    public void SavePersonData(HttpPostedFileBase personPhoto, TestPerson person)
    {

        var dataObject = Request.Form["person"];
        var serializer = new JavaScriptSerializer();
        TestPerson personReport = serializer.DeserializeObject(dataObject, typeOf(TestPerson));

        System.Console.WriteLine("dummy line");
    }

我的.cshtml页面:

<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.2.2/jquery.min.js"></script>
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/js/bootstrap.min.js"></script>
<script src="http://ajax.aspnetcdn.com/ajax/knockout/knockout-3.3.0.js"></script>

<div id="testKo">
    Name: <input type="text" data-bind="value: Name"/>
    Place: <input type="text" data-bind="value: Place" /><br />
    Age: <input type="text" data-bind="value: Age" /><br />

    <input type="button" id="submitForm" class="btn-lg" value="Submit" />
</div>

<form id="photoForm" method="post" enctype="multipart/form-data">
    <input type="hidden" id="personId" name="person" />
    <input type="file" id="personPhotoId" name="personPhoto" />
</form>

<script>
    $(document).ready(function () {

        $(document).ready(function () {

            var element = document.getElementById("testKo");
            ko.applyBindings(viewModel, element);

            $("#submitForm").on("click", function () {

                var vmData = ko.toJSON(viewModel);
                //$("#personId").val({ person: vmData });
                var formData = new FormData($("#photoForm")[0]);

                formData.append("person", vmData, "person");

                //var data = new FormData();
                //data.append("vmData", vmData);
                //data.append("photo", $("#personPhoto").get(0).files[0])

                $.ajax({
                    type: "POST",
                    url: '@Url.Action("SavePersonData", "Home")',
                    //cache: false,
                    //async: false,
                    contentType: false,
                    processData: false,
                    enctype: "multipart/form-data",
                    dataType: "json",
                    //data: {
                    //    testPerson: vmData
                    //},
                    //data: {
                    //    personPhoto: formData,
                    //    person: vmData
                    //},
                    data: formData,
                    success: function (data) {
                        alert("success!");
                    }
                });
            });
        });

        function generatedViewModel() {
            var self = this;

            self.Id = ko.observable("1");
            self.Name = ko.observable();
            self.Place = ko.observable();
            self.Age = ko.observable();
            self.File = ko.observable();
        }
        var viewModel = new generatedViewModel();

    });
</script>

有什么方法可以使用此代码在我的控制器操作方法中接收这两个参数?或者我需要以任何方式调整它吗?我只需要将包括文件/图像上传数据和非表格数据对象在内的所有内容发送到我的控制器方法。

非常感谢任何帮助!

2 个答案:

答案 0 :(得分:0)

嗯,对此最好的解决方案是在另一个内部进行2个同步的ajax调用。发送&#34; personPhoto&#34;首先将数据转换为字节数组并将其设置为Session变量,并在此ajax调用的成功事件中,对main方法进行另一个ajax调用,在该方法中处理Form / Model数据并使用缓存的会话对象进行&#34; personPhoto&#34 ;.

用以下内容替换你的ajax代码:

            var data = new FormData($("#photoForm").get(0));

            $.ajax({
                type: "POST",
                url: '@Url.Action("CacheUploads", "Home")',
                data: data,
                processData: false,
                contentType: false,
                dataType: "json",
                success: function () {

                    var vmData = ko.toJS(viewModel);

                    $.ajax({
                        type: "POST",
                        url: '@Url.Action("SavePersonData", "Home")',
                        data: { person: vmData },
                        success: function (data) {

                        }
                    });
                }
            });

用以下内容替换您的控制器方法:

    public void SavePersonData(TestPerson person)
    {
        // You no longer need to deserialize as you'll have data properly mapped to the TestPerson object.

        //var dataObject = Request.Form["person"];
        //var serializer = new JavaScriptSerializer();
        //object personReport = serializer.DeserializeObject(dataObject);

        person.File = (byte[])Session["UploadedPhoto"];

        System.Console.WriteLine("dummy line");
    }

    public JsonResult CacheUploads(HttpPostedFileBase personPhoto)
    {
        byte[] photoAsBytes = null;

        using (var binaryReader = new BinaryReader(personPhoto.InputStream))
        {
            photoAsBytes = binaryReader.ReadBytes(personPhoto.ContentLength);
        }

        Session.Add("UploadedPhoto", photoAsBytes);

        return Json(new { success = true }, JsonRequestBehavior.AllowGet);
    }

没有其他方法可以在单个ajax调用中一起处理FormData和Non-Form ViewModel数据。这是我从解决方案的角度来看最接近的。祝你好运!

答案 1 :(得分:-1)

使用它。

static.yourdomain.com