微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

在 .net-core 中选择 onchange 到 Controller's Action

如何解决在 .net-core 中选择 onchange 到 Controller's Action

我有一个按类型分类的对象列表,所以我的视图中有一个“类型”选择

<form asp-action="Index" asp-route-type=" ? selectedTypeId ? ">
    <select asp-items="ViewBag.Types" onchange="this.form.submit()"></select>
</form>

我的控制器的动作是

public async Task<IActionResult> Index([FromQuery(Name = "type")] int type) {
   // filter the object list by type 
}

如何使用选择正确地重新加载视图?

我看到有人提出了 Ajax 解决方案,这将是一个很好的解决方案,但是我的问题是,当我更改产品类型时,不仅产品列表会更改,产品特性(列)也会更改,创建按钮也应更改只创建那种类型的产品,所以,它改变了所有的视图行为。我宁愿重新加载所有视图而不是其中的一部分。

解决方法

如何使用选择正确地重新加载视图?

看来是想根据DropDownList的selected值过滤数据,在我看来,我更喜欢用Partial View来展示过滤后的数据,用JQuery Ajax来更新内容。请参考以下示例:

创建一个 ProductViewModel 视图模型:

public class ProductViewModel
{
    public int ProductId { get; set; }
    public string ProductName { get; set; }
    public string ProductType { get; set; }
}

控制器方法:

    //ProductIndex Page
    public IActionResult ProductIndex()
    {
        //get all the initial data from the repository
        var productlist = _repo.GetProductViewModels();

        //set select items for the DropDownList
        ViewBag.Types = _repo.GetProductViewModels().Select(c=>c.ProductType).Distinct().ToList().Select(c => new SelectListItem { Text = c,Value = c }).ToList();
        return View(productlist);
    }

    //based on the type filter data and return partial view.
    public IActionResult ShowProduct(string type)
    {
        var productlist = _repo.GetProductViewModels().ToList();

        //based on the type to filter data.
        if (!string.IsNullOrEmpty(type) && type != "all")
        {
            productlist = productlist.Where(c => c.ProductType == type).ToList();
        }

        //required using Microsoft.AspNetCore.Mvc.Rendering;
        ViewBag.Types = _repo.GetProductViewModels().Select(c => new SelectListItem { Text = c.ProductType,Value = c.ProductType }).Distinct().ToList();
        return PartialView("_ShowProductPartialView",productlist );
    }

创建一个 _ShowProductPartialView.cshtml 局部视图来显示产品列表:

@model IEnumerable<netcore5.Models.ProductViewModel>
 
<table class="table">
    <thead>
        <tr>
            <th>
                @Html.DisplayNameFor(model => model.ProductId)
            </th>
            <th>
                @Html.DisplayNameFor(model => model.ProductName)
            </th>
            <th>
                @Html.DisplayNameFor(model => model.ProductType)
            </th>
            <th></th>
        </tr>
    </thead>
    <tbody>
@foreach (var item in Model) {
        <tr>
            <td>
                @Html.DisplayFor(modelItem => item.ProductId)
            </td>
            <td>
                @Html.DisplayFor(modelItem => item.ProductName)
            </td>
            <td>
                @Html.DisplayFor(modelItem => item.ProductType)
            </td>
            <td>
                @Html.ActionLink("Edit","Edit",new { /* id=item.PrimaryKey */ }) |
                @Html.ActionLink("Details","Details",new { /* id=item.PrimaryKey */ }) |
                @Html.ActionLink("Delete","Delete",new { /* id=item.PrimaryKey */ })
            </td>
        </tr>
}
    </tbody>
</table>

ProductIndex 页面中的代码:

@model IEnumerable<netcore5.Models.ProductViewModel>

@{
    ViewData["Title"] = "ProductIndex";
}

<h1>ProductIndex</h1>
 
<form asp-action="ProductIndex">
    <select id="ddltype" name="type" asp-items="ViewBag.Types" >
        <option value="all">All Type</option>
    </select>
    <div id="productcontainer"> 
        <partial name="_ShowProductPartialView.cshtml" model="@Model" />
    </div> 
</form>


@section Scripts{ 
<script>  
    $(function () {
        $("#ddltype").change(function () {
            $.ajax({
                type: "Get",url: "/Home/ShowProduct?type=" + $(this).val(),//remember change the controller to your owns.  
                success: function (data) {
                    $("#productcontainer").html("");
                    $("#productcontainer").html(data);
                },error: function (response) {
                    console.log(response.responseText);
                }
            });
        });
    });
</script>
}

然后,结果如下:

enter image description here

[注意] 使用上述方法,主页面不会更新,只刷新局部视图。

如果要使用<select>元素onchange事件,可以在Form中添加Submit按钮,在onchange事件中,点击{{ 1}} 按钮,它将提交表单。然后,您可以根据类型过滤数据。

将上面的代码修改如下:在ProductIndex.cshtml页面,去掉JQuery脚本,添加一个Submit按钮,为Submit元素添加name属性,然后,在select select事件中,点击提交按钮:

onchange

控制器中的代码:

@model IEnumerable<netcore5.Models.ProductViewModel>

@{
    ViewData["Title"] = "ProductIndex";
}

<h1>ProductIndex</h1>
 
<form asp-action="ProductIndex">
    <select id="ddltype" name="type" asp-items="ViewBag.Types" onchange="document.getElementById('btnSubmit').click();" >
        <option value="all">All Type</option>
    </select>
    <div id="productcontainer"> 
        <partial name="_ShowProductPartialView.cshtml" model="@Model" />
    </div>

    <input id="btnSubmit" hidden type="submit" value="submit" />
</form>

结果如下:

enter image description here

正如我们所见,主页面将刷新。所以,我更喜欢使用 JQuery Ajax 来加载局部视图。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。