【问题标题】:Passing filter as object from View to Controller将过滤器作为对象从视图传递到控制器
【发布时间】:2017-04-12 16:46:15
【问题描述】:

我正在尝试在 MVC 控制器中实现过滤和分页。我使用 ViewBag 将信息传递给 View。

过滤器类是这样的:

public class Criteria
{
    public int? SnapshotKey { get; set; }
    public string Delq { get; set; }

    public override string ToString()
    {
        string[] Ares = new string[2];
        if (SnapshotKey.HasValue)
            Ares[0] = "SnapshotKey=" + SnapshotKey.ToString();
        if (!String.IsNullOrWhiteSpace(Delq))
            Ares[1] = "Delq=" + Delq;

        return String.Join("&", Ares.Where(s => !string.IsNullOrWhiteSpace(s)));
    }
}

我的控制器方法是这样的:

public ActionResult Search(Criteria filter, string filterString, 
                           int pageNo = 1, int pageSize = 5)
{
    using (DemoDBEntities db = new DemoDBEntities())
    {
        var list = db.BMH.AsQueryable();
        if (filter != null)
        {
            if (filter.SnapshotKey.HasValue)
                list = list.Where(r => r.SnapshotKey == filter.SnapshotKey.Value);
            if (!String.IsNullOrWhiteSpace(filter.Delq))
                list = list.Where(r => r.Delq == filter.Delq);
        }
        list = list.OrderBy(r=>r.SnapshotKey).Skip((pageNo - 1) * pageSize).Take(pageSize);
        ViewBag.Criteria = filter.ToString();
        ViewBag.CriteriaString = filterString;
        ViewBag.PageNo = pageNo;
        return View(list.ToList());
    }
}

AFAIK,我无法将 ViewBag 作为对象传递给控制器​​,这就是我使用 filter.ToString() 来存储当前过滤器的原因。

在视图中,我有以下链接可以转到特定页面,同时保留当前过滤器。

@Html.ActionLink("Next Page", "Search", new { filter = ViewBag, filterString = ViewBag.CriteriaString, pageNo = ViewBag.PageNo + 1, pageSize = 5 })

所以当从 View 返回时,我将当前过滤器作为字符串。现在在控制器中,我需要将字符串转换为 Criteria 类。这是可行的,但我正在寻找一种更体面的方式来做我需要的事情。

【问题讨论】:

  • 为什么不只是new { SnapshotKey = ViewBag.SnapshotKey, Delq = ViewBag.Delq, pageNo = ViewBag.PageNo + 1.... }(你可以通过在你的模型中添加pageNopageSize来进一步简化它,所以POST方法就是public ActionResult Search(Criteria filter)
  • 我试图不依赖于每个文件名,并且如果我向 Criteria 类添加了一个新字段,我不想更改它。这有意义吗?
  • 好吧,如果你添加一个新字段,那么你需要更新.ToString() 方法(即必须更新一些东西)。您可以随时将pageNopageSize 添加到Criteria 模型中,并将它们包含在ToString() 方法中,然后使用<a href="@Url.Action("Search")?@ViewBag.Criteria>Next Page</a> 生成链接
  • 虽然不是很优雅,但您可以使用反射循环模型中的每个属性来构建查询字符串,这样您就不必对每个属性/值进行硬编码。
  • 对,这解决了控制器部分,但是当用户单击 [下一步] 并且程序返回控制器时,我需要知道当前过滤器是什么(在搜索操作中),以便我可以重建使用 LINQ 的行列表。此外,不使用过滤器类使我每次添加新字段进行过滤时都会更改搜索操作的签名。

标签: c# asp.net-mvc filtering paging


【解决方案1】:

Search() 方法中filterString 的值将是格式为name=value&name=value... 的字符串,因此您可以首先使用String.Split()(在& 字符上)创建name=value 项目数组, 然后再次拆分(在 = 字符上)以获取属性名称和值,但这一切都变得混乱,只需构建整个查询字符串并将其直接绑定到您的 Criteria 模型会更容易.

更改模型以包含所有属性,包括 pageNo 和 pageSize`

public class Criteria
{
    public Criteria() // perhaps add some defaults?
    {
        PageNo = 1;
        PageSize = 5;
    }
    public int? SnapshotKey { get; set; }
    public string Delq { get; set; }
    public int PageNo { get; set; }
    public int PageSize { get; set; }
    .... // see method below
}

然后为了使其灵活并允许您添加更多“条件”属性,使用反射来构建查询字符串

    public string ToQueryString(int pageIncrement)
    {
        List<string> propValues = new List<string>();
        foreach(var prop in GetType().GetProperties())
        {
            var name = prop.Name;
            var value = prop.GetValue(this);
            if (name == "PageNo")
            {
                value == (int)value + pageIncrement;
            }
            if (value != null)
            {
                propValues .Add(String.Format("{0}={1}", name, value));
            }
        }
        return "?" + String.Join("&", propValues);
    }

控制器中的代码将是

public ActionResult Search(Criteria filter)
{
    using (DemoDBEntities db = new DemoDBEntities())
    {
        var list = db.BMH.AsQueryable();
        if (filter != null)
        {
            if (filter.SnapshotKey.HasValue)
                list = list.Where(r => r.SnapshotKey == filter.SnapshotKey.Value);
            if (!String.IsNullOrWhiteSpace(filter.Delq))
                list = list.Where(r => r.Delq == filter.Delq);
        }
        list = list.OrderBy(r => r.SnapshotKey).Skip((filter.PageNo - 1) * pageSize).Take(filter.PageSize);
        ViewBag.Criteria = filter;
        return View(list.ToList());
    }
}

然后在视图中

<a href="@Url.Action("Search")@ViewBag.Criteria.ToQueryString(-1)">Previous</a>
<a href="@Url.Action("Search")@ViewBag.Criteria.ToQueryString(1)">Next</a>

还请注意,您可以只使用

@Html.ActionLink("Next", "Search", (yourAssembly.Criteria)ViewBag.Criteria)

假设Criteria 仅包含简单属性,这意味着不需要ToQueryString() 方法。但是,您需要在使用ActionLink() 之前增加/减少PageNo 属性的值,例如

@{ var criteria = (yourAssembly.Criteria)ViewBag.Criteria; }
@{ criteria.PageNo = @criteria.PageNo - 1; }
@Html.ActionLink("Previous", "Search", criteria)
@{ criteria.PageNo = @criteria.PageNo + 2; }
@Html.ActionLink("Next", "Search", criteria)

【讨论】:

  • 干得好。谢谢!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2017-03-31
  • 1970-01-01
  • 1970-01-01
  • 2016-05-28
  • 2019-07-06
  • 2018-10-29
相关资源
最近更新 更多