ASP MVC 5 - 使用级联DropDown列表动态添加多个部分视图

时间:2017-08-22 20:48:13

标签: c# asp.net-mvc html.dropdownlistfor asp.net-mvc-partialview cascadingdropdown

我面临的问题类似于这篇旧帖cascading dropdown for dynamically added row。我正在使用" BeginCollectionItemCore" NuGet包用于设置一组部分视图,因此我的控件具有唯一的ID。问题是,我只能得到第一个部分视图来响应下拉列表更改。随后的下拉列表不会级联。我已尝试使用部分视图和主视图中的脚本,但两者都具有相同的最终结果,只有第一个部分将级联。这是我的代码......

主视图HTML:

@model IEnumerable<RCRTCWA.DATA.DAL.tbl_RCRTimeCards>

@{
    ViewBag.Title = "Index";
}

<h2>Time Card</h2>

@using (Html.BeginForm())
{
    <div id="TimeCardLines">
        @foreach (var item in Model)
        {
            Html.RenderPartial("_TimeCardRow", item);
        }

    </div>
}

@Html.ActionLink("Add more time...", "_TimeCardRow", new { ViewContext.FormContext.FormId }, new { id = "addTime"})

主视图脚本:

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

        $("#addTime").click(function () {
            $.ajax({
                url: this.href,
                cache: false,
                success: function (html) { $("#TimeCardLines").append(html); }

            });
            return false;
        });

        //var index = "";
        $(".timecardrow").focusin(function () {
            var ti = $(this).find("[name='timecardrows.index']");
            var index = ti.val();

            $("#timecardrows_" + index + "__HRS_EndTime").timepicker({
                defaultTime: 'now',
                minTime: '6:00am',
                maxTime: '7:00pm'
            });
            $("#timecardrows_" + index + "__HRS_StartTime").timepicker({
                defaultTime: 'now',
                minTime: '6:00am',
                maxTime: '7:00pm'
            });

            //$("#.Line_TimeCard").ajaxSuccess(function () {
            //    $.getJSON("/TimeCard/AddTimeCardRow/", $("#.Success").html(data).show());
            //});

            $("#timecardrows_" + index + "__WOTicketNo").change(function () {
                var tktid = $(this).val();
                $("#timecardrows_" + index + "__WOTicketRepLineNo").empty();

                $.ajax({
                    url: "/TimeCard/GetRepairLines/",
                    data: { ticket: tktid },
                    cache: false,
                    type: "POST",
                    success: function (data) {
                        $.each(data, function (i, data) {
                            $("#timecardrows_" + index + "__WOTicketRepLineNo").append('<option value="' + data.Value + '">' + data.Text + '</option>');
                        });
                    },
                    error: function (response) {
                        alert("Error : " + response);
                    }
                });
                GetCarNumber(tktid);
            });

            $("#timecardrows_" + index + "__WOTicketRepLineNo").change(function () {
                var line = $("#timecardrows_" + index + "__WOTicketRepLineNo").val();
                $("#timecardrows_" + index + "__WOTicketLaborLineNo").empty();

                $.ajax({
                    url: "/TimeCard/GetLineDetails/",
                    data: { lineid: line },
                    cache: false,
                    type: "POST",
                    success: function (data) {
                        $.each(data, function (i, data) {
                            $("#timecardrows_" + index + "__WOTicketLaborLineNo").append('<option value="' + data.Value + '">' + data.Text + '</option>');
                        });
                    },
                    error: function (response) {
                        alert("error : " + response);
                    }
                });
                return false;
            }).change();



            function GetCarNumber(ticket) {
                $.ajax({
                    url: "/TimeCard/GetCarNumber/",
                    data: { ticket: ticket },
                    cache: false,
                    type: "POST",
                    success: function (data) {
                        $("#timecardrows_" + index + "carNo").html(data).show();
                    },
                    error: function (response) {
                        alert("Error : " + response);
                    }
                });
            }
        });
    });

</script>

部分查看HTML:

@using HtmlHelpers.BeginCollectionItem
@model RCRTCWA.DATA.DAL.tbl_RCRTimeCards

<div class="timecardrow">
    @using (Html.BeginCollectionItem("timecardrows"))
    {
        <div class="form-horizontal">
            <hr />
            @Html.ValidationSummary(true, "", new { @class = "text-danger" })

            <table>
                <tr>
                    <th class="col-sm-1">
                        Ticket Number
                    </th>
                    <th class="col-sm-1">
                        Car Number
                    </th>
                    <th class="col-sm-1">
                        Repair Line / Description
                    </th>
                    <th class="col-sm-1">
                        Labor Line / Description
                    </th>
                    <th class="col-sm-1">
                        Start Time
                    </th>
                    <th class="col-sm-1">
                        End Time
                    </th>
                    <th class="col-sm-1">
                        Line Complete?
                    </th>
                </tr>
                <tr>
                    <td class="form-group">
                        <div class="col-sm-1 tickets">
                            @Html.DropDownListFor(model => model.WOTicketNo, (SelectList)ViewData["TicketsList"], "Select one...", new { @class = "ticketddl" } )
                            @Html.ValidationMessageFor(model => model.WOTicketNo, "", new { @class = "text-danger" })
                        </div>
                    </td>
                    <td class="form-group">
                        <div class="col-sm-1 cars">
                            <div id="carNo"></div>
                            @Html.HiddenFor(model => model.CarNo)
                            @Html.ValidationMessageFor(model => model.CarNo, "", new { @class = "text-danger" })
                        </div>
                    </td>
                    <td class="form-group">
                        <div class="col-sm-1 replines">
                            @Html.DropDownListFor(model => model.WOTicketRepLineNo, new SelectList(string.Empty, "Value", "Text"), "Select one...", new { @class = "repairddl" })
                        </div>
                    </td>
                    <td class="form-group">
                        <div class="col-sm-1 laborlines">
                            @Html.DropDownListFor(model => model.WOTicketLaborLineNo, new SelectList(string.Empty, "Value", "Text"), "Select one...", new { @class = "lablineddl" })
                        </div>
                    </td>
                    <td class="form-group">
                        <div class="col-sm-1 starttime">
                            @Html.EditorFor(model => model.HRS_StartTime, new { @class = "start" })
                            @Html.ValidationMessageFor(model => model.HRS_StartTime, "", new { @class = "text-danger" })
                        </div>
                    </td>
                    <td class="form-group">
                        <div class="col-sm-1 endtime">
                            @Html.EditorFor(model => model.HRS_EndTime, new { @class = "end" })
                            @Html.ValidationMessageFor(model => model.HRS_EndTime, "", new { @class = "text-danger" })
                        </div>
                    </td>
                    <td class="form-group">
                        <div class="col-sm-1 completed">
                            @Html.EditorFor(model => model.Completed)
                            @Html.ValidationMessageFor(model => model.Completed, "", new { @class = "text-danger" })
                        </div>
                    </td>
                    @*<td class="form-group">
                        <div class="col-sm-1">
                            <input type="submit" value="Submit Line" class="btn btn-default" />
                        </div>
                        <div id="success" class="alert-danger">

                        </div>
                    </td>*@
                </tr>

            </table>

        </div>
    }
</div>

目前我在主视图中有脚本,并且在用户与局部视图交互时尝试获取部分视图控件的索引部分。我需要一种更好的方法来处理这个问题,并且需要让级联下拉列表正常工作。

我更希望在主视图中使用脚本(如果可能)以使事情更简单。

2 个答案:

答案 0 :(得分:1)

您不需要使用连接的id字符串在下拉列表中绑定更改事件处理程序,这很容易出错(错别字等)并且不易读取/维护。对于级联下拉方案,您关心的只是更新同一行中的第二个select元素。 jQuery有一些方便的方法,如closestfind,这将使我们的生活更轻松。

为了让未来的读者更容易,我将假设您的第一个SELECT元素是呈现国家/地区列表并且有一个css类&#34; countrySelect&#34;第二个是针对所选国家的州,并且有css类&#34; statesSelect&#34;并且两者都在同一个表行(<tr>)。

绑定change事件时,请确保使用jQuery on来执行此操作。这将启用DOM中当前和未来元素的绑定

$("#TimeCardLines").on("change","SELECT.countrySelect",function (e) {
    var _this = $(this);
    var v = _this.val();

    // to do  :Change below url variable value as needed for your code
    var urlToGetSecondDropDownData = "/Home/GetStates?countryId"+v;

    $.getJSON(urlToGetSecondDropDownData,
        function (res) {
            var items = "";
            $.each(res,
                function(index, item) {
                    items += "<option value='" + item.Value + "'>" 
                                               + item.Text + "</option>";
                });
            _this.closest("tr")                // Get the same table row
                 .find("SELECT.statesSelect")  // Find the second dropdown
                 .html(items);                 // update the content of it

        });
});

假设您有一个GetStates操作方法,该方法接受countryId参数并将相应的状态作为SelectListItem列表返回(构建第二个下拉列表所需的数据)。如下所示,

public ActionResult GetStates(int countryId)
{
    var states =db.States.Where(f => f.CountryId == countryId)
        .Select(f => new SelectListItem() { Value = f.Id.ToString(), 
                                            Text = f.Name})
        .ToList();

    return Json(states, JsonRequestBehavior.AllowGet);

}

答案 1 :(得分:0)

   //first dropdowlist change event 
  $("#countryId").change(function () {
    var id = this.value;
    if (id == "") {
        id = "0";
    }
    $.get('/Home/GetStates/' + id,
          function (data) {

              $('#stateId').find('option').remove()
              $(data).each(
                  function (index, item) {
                      $('#stateId').append('<option value="' + item.Value + '">' + item.Text + '</option>')
                  });
          }
);
});




 public ActionResult GetStates(int id)
    {

        List<SelectListItem> StatesList= new List<SelectListItem>();
        StatesList.Add(new SelectListItem { Text = "---Select State---", Value = "0" });
        var getStateCollection= (from f in _ent.States
                              where f.CountryId == id && f.DeletedDate == null
                              select new { f.Id, f.Name}).ToList();


        foreach (var item in getStateCollection)
        {
            StatesList.Add(new SelectListItem { Text = item.Name.ToString(), Value = item.Id.ToString() });
        }
        return Json(StatesList, JsonRequestBehavior.AllowGet);
    }