使用knockout和mvc

时间:2015-06-02 11:59:43

标签: knockout-mvc knockout-3.0

@model NewDemoApp.Models.DemoViewModel
@{
    ViewBag.Title = "Home Page";
}


@*<script src="@Url.Content("~/Scripts/jquery-1.9.1.min.js")" type="text/javascript"></script>*@
<script src="http://code.jquery.com/jquery-1.11.3.min.js"></script>
<script src="http://code.jquery.com/jquery-migrate-1.2.1.min.js"></script>

<script src="@Url.Content("~/Scripts/knockout-3.3.0.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>
<script type="text/javascript">
    var viewModel; 

    var compViewModel, userViewModel;

    $(document).ready(function () {

        $(".wizard-step:visible").hide();
        $(".wizard-step:first").show(); // show first step
        $("#back-step").hide();


        var result = @Html.Raw(Json.Encode(Model));

        var viewModel = new DemoViewModel(result.userViewModel);

        //viewModel.userViewModel.FirstName = result.userViewModel.FirstName;
        //viewModel.userViewModel.LastName = result.userViewModel.LastName;
        //viewModel.userViewModel.State = result.userViewModel.State;
        //viewModel.userViewModel.City = result.userViewModel.City;
        ko.applyBindings(viewModel);

    });

    var userVM = function(){

        FirstName = ko.observable(),
        LastName = ko.observable(),
        State = ko.observable(),
        City = ko.observable()
    };

    function DemoViewModel(data) {

           var self = this;



           self.userViewModel = function UserViewModel(data) {
               userVM.FirstName = data.FirstName;
               userVM.LastName = data.LastName;
               userVM.State = data.State;
               userVM.City = data.City;
           }

           self.Next = function () {
               var $step = $(".wizard-step:visible"); // get current step
               var form = $("#myFrm");

               var validator = $("#myFrm").validate(); // obtain validator
               var anyError = false;
               $step.find("input").each(function () {
                   if (!validator.element(this)) { // validate every input element inside this step
                       anyError = true;
                   }

               });

               if (anyError)
                   return false; // exit if any error found



               if ($step.next().hasClass("confirm")) { // is it confirmation?
                   $step.hide().prev(); // hide the current step
                   $step.next().show(); // show the next step
                   $("#back-step").show();
                   $("#next-step").hide();
                   //$("#myFrm").submit();
                   // show confirmation asynchronously
                   //$.post("/wizard/confirm", $("#myFrm").serialize(), function (r) {
                   //    // inject response in confirmation step
                   //    $(".wizard-step.confirm").html(r);
                   //});

               }
               else {
                   if ($step.next().hasClass("wizard-step")) { // is there any next step?
                       $step.hide().next().fadeIn();  // show it and hide current step
                       $("#back-step").show();   // recall to show backStep button
                       $("#next-step").show();
                   }
               }
           }

           self.Back = function () {

               var $step = $(".wizard-step:visible"); // get current step
               if ($step.prev().hasClass("wizard-step")) { // is there any previous step?
                   $step.hide().prev().fadeIn();  // show it and hide current step

                   // disable backstep button?
                   if (!$step.prev().prev().hasClass("wizard-step")) {
                       $("#back-step").hide();
                       $("#next-step").show();
                   }
                   else {
                       $("#back-step").show();
                       $("#next-step").show();
                   }
               }
           }

           self.SubmitForm = function (formElement) {             

                   $.ajax({
                   url: '@Url.Content("~/Complaint/Save")',
                   type: "POST",
                   data: ko.toJS(self),
                   done: function (result) {                       
                       var newDiv = $(document.createElement("div"));
                       newDiv.html(result);
                   },
                   fail: function (err) {

                       alert(err);
                   },
                   always: function (data) {

                       alert(data);
                   }
               });
           }

          self.loadData = function () {

               $.get({
                   url: '@Url.Content("~/Complaint/ViewComplaint")',

                   done: function (data) {
                       debugger;
                       alert(data);

                       self.compViewModel(data);
                       self.userViewModel(data);

                   },
                   fail: function (err) {
                       debugger;
                       alert(err);
                   },
                   always: function (data) {
                       debugger;
                       alert(data);
                   }
               });


           }

       }







</script>




<form class="form-horizontal" role="form" id="myFrm">
    <div class="container">
        <div class="row">
            <div class="col-md-3">

            </div>
            <div class="col-md-6">
                <div class="wizard-step">

                </div>
                <div class="wizard-step" >
                    <h3> Step 2</h3>
                    @Html.Partial("UserView", Model.userViewModel)
                    <div class="col-md-3"></div>
                    <div class="col-md-6">
                        <input type="submit" id="submitButton" class="btn btn-default btn-success" value="Submit" data-bind="click: SubmitForm" />

                    </div>
                    <div class="col-md-3"></div>
                </div>
                <div class="wizard-step">
                    <h3> Step 3</h3>
                </div>
                <div class="wizard-step confirm">
                    <h3> Final Step 4</h3>

                </div>
            </div>

            <div class="col-md-3"></div>
        </div>
        <div class="row">
            <div class="col-md-3"></div>
            <div class="col-md-6">
                <input type="button" id="back-step" class="btn btn-default btn-success" value="< Back" data-bind="click: Back" />
                <input type="button" id="next-step" class="btn btn-default btn-success" value="Next >" data-bind="click: Next" />
            </div>
            <div class="col-md-3"></div>
        </div>
    </div>
</form>

我能够从控制器获取数据并使用knockout绑定它。部分视图从控制器加载数据。但是,在提交更新的数据时,我没有获得更新的数据,而是获得无法从空引用访问“FirstName”属性的错误。我只需要指出我出错的地方,特别是创建ViewModel并使用它们的正确方法。

1 个答案:

答案 0 :(得分:0)

当您在self.SubmitForm函数中提交表单时,您将传递从Knockout视图模型转换的Json对象。

因此,请确保正确地在所有输入标记中提供data-bind属性。如果您使用的是Razor语法,请在输入标记的Html属性中使用data_bind

检查KO的双向绑定是否正常。我无法确定您是否未共享部分视图Razor代码。

Refer- http://knockoutjs.com/documentation/value-binding.html

在Chrome中,您可以在javascript开发者控制台的“网络”标签中查看您要提交的数据。您要发布的Json数据和您在控制器方法中期望的ViewModel数据结构应该匹配。

您还可以将参数更改为FormCollection formCollection,并检查发布时来自浏览器的数据。