首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >用MVC3敲除:敲除模型不会发布收集数据?

用MVC3敲除:敲除模型不会发布收集数据?
EN

Stack Overflow用户
提问于 2012-03-23 15:27:43
回答 1查看 5K关注 0票数 3

回答:替换了这一行:

代码语言:javascript
复制
self.identifiers.push(new Identifier(self.identifierToAdd(), self.selectedIdentifierTypeValue()));

用这一行:

代码语言:javascript
复制
self.identifiers.push({ Key: self.identifierToAdd(), Value: self.selectedIdentifierTypeValue()});

post请求现在正在正确地发送收集数据。然而,这并不能解决MVC操作没有收到它的事实,但是这个问题已经足够大了。

当发布到一个操作时,我似乎无法从我的模型的集合属性中获取数据,而将数据输入到我的MVC模型中。如果我从下面警告ko.toJSON我的identifiers()属性,它正确地显示了所有数据,但是当我尝试通过正常回发提交数据时(下面的操作只是采取EquipmentCreateModel ),它看起来如下所示:

标识符是空的,当我查看标识符的ModelState错误时,它说它是cannot convert String to Dictionary<Guid, string>。我做错了什么?我认为MVC3可以自动将JSON转换为对象,就像它对BuildingCodeRoom属性所做的那样?

此外,为什么我的字符串数据在上面的图片有转义引号?

编辑:--如果我查看post数据,标识符将显示为空数组(identifiers: [{}])。我在保存方法中尝试了jsoning标识符,如下所示:

代码语言:javascript
复制
self.identifiers = ko.toJSON(self.identifiers());

这将导致请求数据不为空,如下所示:

代码语言:javascript
复制
identifiers:"[{\"Value\":\"sdfsd\",\"Key\":\"4554f477-5a58-4e81-a6b9-7fc24d081def\"}]"

但是,在调试操作时也会出现同样的问题。我还尝试了整个模型(如knockoutjs submit with ko.utils.postJson issue中所概述的):

代码语言:javascript
复制
ko.utils.postJson($("form")[0], ko.toJSON(self));

但是这给出了一个.NET错误,它表示Operation is not valid due to the current state of the object.,从查看请求数据来看,它看起来像是被JSON化了两次,因为每个字母或字符在HttpCollection中都是它自己的值,这是因为.NET默认只允许1000max ('Operation is not valid due to the current state of the object' error during postback)。

使用$.ajax方法提交数据,一切都很好:

代码语言:javascript
复制
 $.ajax({
            url: location.href, 
            type: "POST",
            data: ko.toJSON(viewModel),
            datatype: "json",
            contentType: "application/json charset=utf-8",
            success: function (data) { alert("success"); }, 
            error: function (data) { alert("error"); }
        });

但是由于其他原因,我不能使用$.ajax方法,所以我需要它在正常的文章中工作。为什么我可以在ajax请求中toJSON整个viewModel,它可以工作,但是在正常回发中它会分割它,当我不这样做的时候,所有的引号都会在发送的JSON中转义。

这是我的ViewModel:

代码语言:javascript
复制
public class EquipmentCreateModel
{
//used to populate form drop downs
public ICollection<Building> Buildings { get; set; }
public ICollection<IdentifierType> IdentifierTypes { get; set; }

[Required]
[Display(Name = "Building")]
public string BuildingCode { get; set; }

[Required]
public string Room { get; set; }

[Required]
[Range(1, 100, ErrorMessage = "You must add at least one identifier.")]
public int IdentifiersCount { get; set; } //used as a hidden field to validate the list
public string IdentifierValue { get; set; } //used only for knockout viewmodel binding

public IDictionary<Guid, string> Identifiers { get; set; }
}

然后是我的退出脚本/视图模型:

代码语言:javascript
复制
<script type="text/javascript">
// Class to represent an identifier
function Identifier(value, identifierType) {
    var self = this;
    self.Value = ko.observable(value);
    self.Key = ko.observable(identifierType);
}

// Overall viewmodel for this screen, along with initial state
function AutoclaveCreateViewModel() {
    var self = this;

    //MVC properties
    self.BuildingCode = ko.observable();
    self.room = ko.observable("55");
    self.identifiers = ko.observableArray();
    self.identiferTypes = @Html.Raw(Json.Encode(Model.IdentifierTypes));
    self.identifiersCount = ko.observable();


    //ko-only properties
    self.selectedIdentifierTypeValue = ko.observable();
    self.identifierToAdd = ko.observable("");

    //functionality
    self.addIdentifier = function() {
        if ((self.identifierToAdd() != "") && (self.identifiers.indexOf(self.identifierToAdd()) < 0)) // Prevent blanks and duplicates
        {
            self.identifiers.push(new Identifier(self.identifierToAdd(), self.selectedIdentifierTypeValue()));
            alert(ko.toJSON(self.identifiers()));
        }
        self.identifierToAdd(""); // Clear the text box
    };

    self.removeIdentifier = function (identifier) {
        self.identifiers.remove(identifier);
        alert(JSON.stringify(self.identifiers()));
    };

    self.save = function(form) {
        self.identifiersCount = self.identifiers().length;
        ko.utils.postJson($("form")[0], self);
    };
}
    var viewModel = new EquipmentCreateViewModel();
    ko.applyBindings(viewModel);
    $.validator.unobtrusive.parse("#equipmentCreation");      
    $("#equipmentCreation").data("validator").settings.submitHandler = viewModel.save;

查看:

代码语言:javascript
复制
@using (Html.BeginForm("Create", "Equipment", FormMethod.Post, new { id="equipmentCreation"}))
{
@Html.ValidationSummary(true)
<fieldset>
    <legend>Location</legend>
    <div class="editor-label">
        @Html.LabelFor(model => model.BuildingCode)
    </div>
    <div class="editor-field">
        @Html.DropDownListFor(model => model.BuildingCode, new SelectList(Model.Buildings, "BuildingCode", "BuildingName", "1091"), "-- Select a Building --", new { data_bind = "value:BuildingCode"})
        @Html.ValidationMessageFor(model => model.BuildingCode)
    </div>
    <div class="editor-label">
        @Html.LabelFor(model => model.Room)
    </div>
    <div class="editor-field">
        @Html.TextBoxFor(model => model.Room, new { @class = "inline width-7", data_bind="value:room"})
        @Html.ValidationMessageFor(model => model.Room)
    </div>
</fieldset>
<fieldset>
    <legend>Identifiers</legend>
    <p>Designate any unique properties for identifying this autoclave.</p>
    <div class="editor-field">
        Add Identifier
        @Html.DropDownList("identifiers-drop-down", new SelectList(Model.IdentifierTypes, "Id", "Name"), new { data_bind = "value:selectedIdentifierTypeValue"})
        @Html.TextBox("identifier-value", null, new { @class = "inline width-15", data_bind = "value:identifierToAdd, valueUpdate: 'afterkeydown'" })
        <button type="submit" class="add-button" data-bind="enable: identifierToAdd().length > 0, click: addIdentifier">Add</button>
    </div>

    <div class="editor-field">
        <table>
            <thead>
                <tr>
                    <th>Identifier Type</th>
                    <th>Value</th>
                    <th></th>
                </tr>
            </thead>
            <!-- ko if: identifiers().length > 0 -->
            <tbody data-bind="foreach: identifiers">

                <tr>
                    <td>
                        <select data-bind="options: $root.identiferTypes, 
                        optionsText: 'Name', optionsValue: 'Id', value: Key">
                        </select>
                    </td>
                    <td><input type="text" data-bind="value: Value"/></td>
                    <td><a href="#" class="ui-icon ui-icon-closethick" data-bind="click: $root.removeIdentifier">Remove</a></td>
                </tr>


            </tbody>
            <!-- /ko -->
            <!-- ko if: identifiers().length < 1 -->
            <tbody>
                <tr>    
                    <td colspan="3"> No identifiers added.</td>
                </tr>
            </tbody>
            <!-- /ko -->
        </table>
        @Html.HiddenFor(x => x.IdentifiersCount, new { data_bind = "value:identifiers().length" })<span data-bind="text:identifiers"></span>
        @Html.ValidationMessageFor(x => x.IdentifiersCount)
    </div>    
</fieldset>
<p>
    <input type="submit" value="Create" />
</p>
}
EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2012-03-23 19:09:08

我想我已经发现了问题,或者至少缩小了问题的范围。可编辑网格示例使用简单的js对象来表示礼物。您使用的是带有子可观测值的标识符对象。看起来,如果我们更新网格示例以使用更复杂的类型,它也会以与您的示例相同的方式中断。这要么是设计的,要么是错误的。

http://jsfiddle.net/SZzVW/9/

我认为唯一的解决方案是编写自己的映射函数来提交表单。

希望这能有所帮助。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/9841955

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档