MVC5 + EF6 + Bootstrap3 (11) 排序、搜索、分頁


文章來源:Slark.NET-博客園 http://www.cnblogs.com/slark/p/mvc5-ef6-bs3-get-started-pagedlist.html 

系列教程:MVC5 + EF6 + Bootstrap3

上一節:MVC5 + EF6 + Bootstrap3 (10) 數據查詢頁面

下一節:MVC5 + EF6 + Bootstrap3 (12) 新建數據

源碼下載:點我下載

目錄

前言

上一節我們做到了如下的一個基礎查詢頁面。本節我們向這個頁面中加入排序、搜索和分頁功能。

排序

從上圖中的地址欄中可以看到這個頁面調用的是Company Controller下的Index Action.因此我們就先打開Controllers文件夾下的CompanyController.cs文件,寫入如下代碼:

 1 using System.Linq;
 2 using System.Web.Mvc;
 3 using SlarkInc.DAL;
 4 using System;
 5 
 6 namespace SlarkInc.Controllers
 7 {
 8     public class CompanyController : Controller
 9     {
10         private CompanyContext db = new CompanyContext();
11         public ViewResult Index(string sortOrder)
12         {
13             ViewBag.FirstNameSortParm = String.IsNullOrEmpty(sortOrder) ? "first_desc" : "";
14             ViewBag.LastNameSortParm = sortOrder == "last" ? "last_desc" : "last";
15             var workers = from w in db.Workers
16                         select w;
17             switch (sortOrder)
18             {
19                 case "first_desc":
20                     workers = workers.OrderByDescending(w => w.FirstName);
21                     break;
22                 case "last_desc":
23                     workers = workers.OrderByDescending(w => w.LastName);
24                     break;
25                 case "last":
26                     workers = workers.OrderBy(w => w.LastName);
27                     break;
28                 default:
29                     workers = workers.OrderBy(w => w.FirstName);
30                     break;
31             }
32             return View(workers.ToList());
33         }
34     }
35 }

第11行,Index Action 的參數sortOrder用來傳遞給Controller排序的信息。sortOrder有4個可能值。這4個值體現在17到31行的switch..case語句中。如果是first_desc則對FirstName用OrderByDescending倒序排列。如果是last_desc則對LastName倒序排列。如果是last則對LastName用OrderBy函數順序排列。如果是空則對FirstName順序排列。第13-14行用Viewbag存儲當前排列的相反排列字符串,用於在View中生成鏈接。第15-16行,用Linq to Entity 從數據庫中選取要排序的數據。第32行將排序好的數據以List的形式傳遞給View。

在對應的View文件~\Views\Company\Index.cshtml中,修改其表頭部分,代碼如下。

<tr>
        <th>
            @Html.ActionLink("First Name", "Index", new { sortOrder = ViewBag.FirstNameSortParm })
        </th>
        <th>
            @Html.ActionLink("Last Name", "Index", new { sortOrder = ViewBag.LastNameSortParm })
        </th>
        <th>@Html.DisplayNameFor(model => model.Sex)</th>
        <th>@Html.DisplayNameFor(model => model.Rating)</th>
</tr>

 黃色的部分為修改后的代碼。我們把以前純文本的表頭改成一個超鏈接,鏈接的參數從ViewBag中獲得。點擊鏈接可以獲得與當前排序相反的排序效果。

 通過瀏覽器查看此頁面。之后點擊LastName鏈接,則按照上面代碼會生成如下頁面:

 

參數sortOrder為last則這個表按lastName順序排列。

搜索

一般查詢頁面都會有搜索功能幫助我們查看自己想要的頁面。下面我們就來實現這個功能。

還是從Controller開始,Index Action的代碼寫成如下形式:

 1 public ViewResult Index(string sortOrder, string searchString)
 2 {
 3     ViewBag.FirstNameSortParm = String.IsNullOrEmpty(sortOrder) ? "first_desc" : "";
 4     ViewBag.LastNameSortParm = sortOrder == "last" ? "last_desc" : "last";
 5     var workers = from w in db.Workers
 6                 select w;
 7     if (!string.IsNullOrEmpty(searchString))  8     {
 9         workers = workers.Where(w => w.FirstName.Contains(searchString) 10                                 || w.LastName.Contains(searchString)); 11     }
12     switch (sortOrder)
13     {
14         case "first_desc":
15             workers = workers.OrderByDescending(w => w.FirstName);
16             break;
17         case "last_desc":
18             workers = workers.OrderByDescending(w => w.LastName);
19             break;
20         case "last":
21             workers = workers.OrderBy(w => w.LastName);
22             break;
23         default:
24             workers = workers.OrderBy(w => w.FirstName);
25             break;
26     }
27     return View(workers.ToList());
28 }

上面代碼中黃色的部分就是我們所添加的。第1行,searchString向Controller傳遞用來搜索的字符串。第7行,字符串不為空時執行搜索。這里的搜索用到了Linq里的Where函數,它通過一個Lambda表達式來篩選出符合條件的數據。這個條件就是9-10行的"或"連接的兩個Contains函數。Contains函數表示其所屬字符串是否包含其參數字符串。整個篩選條件表示返回所有FisrtName或LastName包含篩選字符串的記錄。

 下面看~\Views\Company\Index.cshtml的改動:

@using(Html.BeginForm())
{
      <p>
           Find by name: @Html.TextBox("SearchString","", new { @class = "form-control", @Style = "display:inline" })
           @Html.Submit("Submit")
      </p>
}

在table的上面加入這些代碼,它是一個Form,包含一個輸入框和提交按鈕,這個Form會把輸入框中的搜索字符串提交給當前Controller完成搜索操作。其中的輸入框,加入了class和style對其進行了Bootstrap美化。

通過瀏覽器查看這個頁面,在搜索框中輸入b並且提交,之后會出現如下頁面:

可以看到搜索結果是FirstName或LastName包含b(不分大小寫),的所有記錄。

分頁

數據查詢免不了要分頁。這里我們用一個名為PagedList的插件來實現這個功能。

如下所示,在菜單欄里選擇工具->庫程序包管理->程序包管理控制台。

 之后會在Visual Studio窗口的下面看到一個程序包管理控制台窗口。在其中輸入Install-Package PagedList.Mvc。PagedList就會自動安裝到項目中。窗口如下所示。

Paged List 安裝好后,我們就開始修改代碼,從Controller開始:

 1 using System.Linq;
 2 using System.Web.Mvc;
 3 using SlarkInc.DAL;
 4 using System;
 5 using PagedList;  6 
 7 namespace SlarkInc.Controllers
 8 {
 9     public class CompanyController : Controller
10     {
11         private CompanyContext db = new CompanyContext();
12         public ViewResult Index(string sortOrder, string searchString, string currentFilter, int? page)
13         {
14             ViewBag.CurrentSort = sortOrder; 15             ViewBag.FirstNameSortParm = String.IsNullOrEmpty(sortOrder) ? "first_desc" : "";
16             ViewBag.LastNameSortParm = sortOrder == "last" ? "last_desc" : "last";
17             if(searchString != null) 18             {
19                 page = 1; 20             }
21             else
22             {
23                 searchString = currentFilter; 24             }
25             ViewBag.CurrentFilter = searchString; 26 
27             var workers = from w in db.Workers
28                         select w;
29             if (!string.IsNullOrEmpty(searchString))
30             {
31                 workers = workers.Where(w => w.FirstName.Contains(searchString)
32                                         || w.LastName.Contains(searchString));
33             }
34             switch (sortOrder)
35             {
36                 case "first_desc":
37                     workers = workers.OrderByDescending(w => w.FirstName);
38                     break;
39                 case "last_desc":
40                     workers = workers.OrderByDescending(w => w.LastName);
41                     break;
42                 case "last":
43                     workers = workers.OrderBy(w => w.LastName);
44                     break;
45                 default:
46                     workers = workers.OrderBy(w => w.FirstName);
47                     break;
48             }
49             int pageSize = 3; 50             int pageNumber = (page ?? 1); 51             return View(workers.ToPagedList(pageNumber,pageSize)); 52         }
53     }
54 }

 黃色標記的為需要改動的部分。第5行,需要引用PagedList。第12行currentFilter參數用來在翻頁時保持搜索字符串不丟失。第17到25行的作用是,當有一個新的字符串要搜索,那么翻當前頁就自動變成第一頁,否則,當前頁不變。第49-51行設置每頁3條數據,設置頁數,並將數據以List的形式發送個View。其中(page ?? 1)的意思是如果page為null則給page賦值為1否則,page不為null那么該是多少就是多少。所以能讓默認頁為1.

 1 @model PagedList.IPagedList<SlarkInc.Models.Worker>
 2 @using PagedList.Mvc;
 3 <link href="~/Content/PagedList.css" rel="stylesheet" />
 4            <br />
 5            @using(Html.BeginForm("Index","Company",FormMethod.Get))
 6            {
 7                <p>
 8                    Find by name: @Html.TextBox("SearchString",ViewBag.CurrentFilter as string, new { @class = "form-control", @Style = "display:inline" })
 9                    @Html.Submit("Submit")
10                </p>
11            }
12 <table class="table">
13     <tr>
14         <th>
15             @Html.ActionLink("First Name", "Index", new { sortOrder = ViewBag.FirstNameSortParm })
16         </th>
17         <th>
18             @Html.ActionLink("Last Name", "Index", new { sortOrder = ViewBag.LastNameSortParm })
19         </th>
20         <th>Sex</th>
21         <th>Rating</th>
22     </tr>
23     @foreach (var item in Model)
24     {
25         <tr>
26             <td>
27                 @Html.DisplayFor(modelItem => item.FirstName)
28             </td>
29             <td>
30                 @Html.DisplayFor(modelItem => item.LastName)
31             </td>
32             <td>
33                 @Html.DisplayFor(modelItem => item.Sex)
34             </td>
35             <td>
36                 @Html.DisplayFor(modelItem => item.Rating)
37             </td>
38         </tr>
39     }
40 </table>
41 <br/>
42 Page @(Model.PageCount < Model.PageNumber ? 0 : Model.PageNumber) of @Model.PageCount
43 @Html.PagedListPager(Model, page => Url.Action("Index",
44     new {page, sortOrder = ViewBag.CurrentSort, currentFilter = ViewBag.CurrentFilter}))

 第1行定義了PagedList 類型的model。第2行引用PagedList。第3行,引入,PagedList相關的css文件,這個文件在安裝插件的時候會自動保存到項目里。第5行BeginForm函數的參數有變化,其生成的Form會使用Get方法。Get方法會在url里顯示提交的參數,這樣可以通過url來記錄查詢參數,方便通過復制url給他人來展現自己的查詢結果。第42行用於顯示第幾頁,共幾頁。第43-44行,用於生成每一頁的按鈕。

運行此頁面,當頁面出來時點擊頁面下方的"2"按鈕轉到第2頁。生成的頁面如下所示:

可以看到頁面下方的分頁按鈕和頁數統計,美觀大方。

結尾

嗯,不知不覺,快半夜一點了。加油!

喜歡的話就推薦下吧!

上一節:MVC5 + EF6 + Bootstrap3 (10) 數據查詢頁面

下一節:MVC5 + EF6 + Bootstrap3 (12) 新建數據

作者:Slark.NET

出處:http://www.cnblogs.com/slark/

本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。如有問題或建議,請多多賜教,非常感謝。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM