Ajax.BeginForm和验证

时间:2013-05-19 14:03:37

标签: asp.net-mvc asp.net-mvc-4

客户端验证在Ajax.BeginForm

中对我不起作用

这是我的代码:

<div id="report">
    <div id="projectReport">
        <div >
            @{
                Html.EnableClientValidation();
            }

            @using (Ajax.BeginForm("AnalyticsDates", new AjaxOptions
                {
                    InsertionMode = InsertionMode.Replace,
                    UpdateTargetId = "reportContent"
                }))
            {
                @Html.LabelFor(m => m.StartDate)
                @Html.TextBoxFor(m => m.StartDate, new { id = "start" })
                @Html.ValidationMessageFor(model => model.StartDate)
                @Html.LabelFor(m => m.EndDate)
                @Html.TextBoxFor(m => m.EndDate, new { id = "end" })
                @Html.ValidationMessageFor(model => model.EndDate)
                <input id="btnsearch" type="submit" value=@Titles.Search class="iconHeader"/>
            }
        </div>
    </div>
    <div id="reportContent">
    </div>
</div>

我在web.config页面中启用了验证:

<add key="ClientValidationEnabled" value="true" />   
<add key="UnobtrusiveJavaScriptEnabled" value="true" />

并添加了js文件

<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.js")" type="text/javascript"></script>

与第一个问题相关的第二个问题, 我的行动是

[HttpPost]
        [Authorize(Roles = "XXXReport")]
        public async Task<ActionResult> AnalyticsDates(ReportRequestVM reportRequestVM)
        {
            if (!ModelState.IsValid)
            {
                return View("**MainReports**", reportRequestVM);
            }

            // fill reportRequestVM with data
            return View("**PartialReport**", reportRequestVM);


        }

如果模型有效,我返回一个局部视图,页面看起来很好,否则我返回主视图,带有表单,但在此页面呈现自己两次。问题是,如果客户端验证失败,如何返回带有验证错误的主窗体?

任何帮助将不胜感激, 10x Rony

2 个答案:

答案 0 :(得分:6)

我明白了...... 你应该对结果和查询有部分的看法。

如果失败,您应该返回“http错误请求”并使用以下内容在搜索部分视图上设置验证。

这应该是它的样子:

        @using (Ajax.BeginForm("CloudAnalyticsDates", new AjaxOptions
            {
                InsertionMode = InsertionMode.Replace,
                UpdateTargetId = "reportContent",
                OnFailure = "OnCloudAnalyticsFailure",
                OnBegin = "ValidateForm",


            }))
        {
            @Html.LabelFor(m => m.StartDate)
            @Html.TextBoxFor(m => m.StartDate, new { id = "start" })
            @Html.ValidationMessageFor(model => model.StartDate)
            @Html.LabelFor(m => m.EndDate)
            @Html.TextBoxFor(m => m.EndDate, new { id = "end" })
            @Html.ValidationMessageFor(model => model.EndDate)
            <input id="btnsearch" type="submit" value=@Titles.Search class="iconHeader"/>
        }
    </div>
</div>

<script type="text/javascript">
    $(document).ready(function () {
        $("#datePicker").kendoDatePicker();
        $("#start").kendoDatePicker().data("kendoDatePicker");
        $("#end").kendoDatePicker().data("kendoDatePicker");
    });


    function OnCloudAnalyticsFailure(parameters) {

        $('#projectReport').html(parameters.responseText);
        $('#reportContent').empty();
        CleanValidationError('form');
    }



</script>

并且在服务器上它应该如下所示:

[HttpPost]

    public async Task<ActionResult> CloudAnalyticsDates(ReportRequestVM reportRequestVM)
    {
        if (!ModelState.IsValid)
        {
            Response.StatusCode = (int)HttpStatusCode.BadRequest;
            return PartialView("_ReportQuery", reportRequestVM);
        }


        reportRequestVM.BomTotals = await CommonRequestsHelper.GetBomTotalAnalytics(bomTotalRequest);
        return PartialView("_ProjectReport", reportRequestVM);
    }

答案 1 :(得分:3)

当您的模型状态无效并且您返回包含错误的模型的视图时,服务器发送的响应的http状态为200,表示请求成功。在这种情况下,ajax表单执行您指示它执行的操作,即将返回的响应插入到div repostContent中(您可以通过检查页面的第二次呈现是否在该div内完成来验证)。至于如何获取ajax提交表单与正常表单提交一样,就显示验证消息而言,我还没有找到一个简单的直接方法(也许有人可以在这里指出一个:)给我们)。我通过以下方式解决了这个问题:

  1. 将表单放在部分视图中,呈现为主视图的div(formDiv)。
  2. 在ajax表单上指定了“OnFailure”,将响应文本插入包含原始表单的div(formDiv)
  3. 覆盖OnActionExecuted(因为我有一个主控制器,我的所有其他人都只从一个地方继承),以检查请求IsAjax和ModelState是否无效,我将响应状态更改为4xx。这导致“OnFailure”被触发,用If(!ModelState.Isvalid)分支中返回的形式替换原始表单。