2016-11-28 169 views
0

我試圖在MVC控制器中實現過濾和分頁。我使用ViewBag將信息傳遞給View。將過濾器作爲查看對象傳遞給控制器​​

Filter類是這樣的:

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()); 
    } 
} 

據我所知,我無法通過ViewBag作爲一個對象來控制,這就是爲什麼我用filter.ToString( )來存儲當前的過濾器。

在視圖中,我有以下鏈接轉到特定頁面,同時保留當前過濾器。

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

所以當從View返回時,我得到當前的過濾器作爲字符串。現在在控制器中,我需要將字符串轉換爲Criteria類。這是可行的,但我正在尋找一個更體面的方式來做我所需要的。

+1

爲什麼不只是'新{SnapshotKey = ViewBag.SnapshotKey,Delq = ViewBag.Delq,幫您做生意= ViewBag.PageNo + 1 ...}'(你可以通過在模型中添加'pageNo'和'pageSize'來簡化它,所以POST方法只是'public ActionResult Search(Criteria filter)' –

+0

我試圖不依賴如果我在Criteria類中添加了一個新字段,這是否有意義呢? – FLICKER

+0

好吧,如果你添加一個新字段,那麼你需要更新'.ToString()'方法(即有些東西需要更新)。你總是可以將'pageNo'和'pageSize'添加到'Criteria'模型中,並將它們包含在'ToString()'方法中,然後使用'Next Page' –

回答

1

filterStringSearch()方法的值將是在格式name=value&name=value...一個字符串,以便可以首先使用String.Split()(在&字符)來創建的name=value項的數組,然後再次分裂(在=字符)以獲得屬性名稱和值,但是這一切都變得混亂,只是構建整個查詢字符串並讓它直接綁定到您的模型上會更容易。

改變模型包括所有屬性,包括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) 
+0

幹得好,謝謝! – FLICKER

相關問題