解决EasyUI-Datagrid和LinqToEntity结合应用时排序问题

时间:2023-03-10 07:05:28
解决EasyUI-Datagrid和LinqToEntity结合应用时排序问题

我们在做WEB页面时,时常会选择JQuery框架的Datagrid,如Ext、EasyUI、Flexigrid,数据访问则采用LinqToSQL或LinqToEntity。UI用Jquery框架的目的是代码简洁,较好的用户体验,漂亮的CSS;数据访问采用LinqToSQL或LinqToEntity的优点是开发便捷,将大部分的时间投入到业务中,而不是Ado.net的数据操作上。

下图是UI和后台的交互简略图,UI将json发送到业务层,业务层将json数据包装成查询对象,然后进行数据访问,获取数据结果集后,再包装成json对象返回给UI,然后UI解析和展现。
解决EasyUI-Datagrid和LinqToEntity结合应用时排序问题
这里我就以EasyUI的Datagrid和LinqToEntity为例。见效果图:
解决EasyUI-Datagrid和LinqToEntity结合应用时排序问题
难点是当需要排序时,LinqToEntity是通过对象的属性指定排序字段,而UI传过来的是字段名,要将排序的字符串转换为Lambda的排序。
分析EasyUI的Datagrid在ajax操作时传递了4个参数,分别是:Page(第几页),Rows(每页记录数),Sort(排序的字段名),Order(排序方式),为此我构建了Datagrid的参数类:

  1. public class DatagridParameters {
  2. public int Page { get; set; }
  3. public int Rows { get; set; }
  4. public string Sort { get; set; }
  5. public string Order { get; set; }
  6. }

参数的包装我是通过一般处理程序来处理,也可以是WCF或其他,哪里处理不是本文的重要之处

  1. //获取参数
  2. DatagridParameters dgParameters = new DatagridParameters {
  3. Page = Convert.ToInt32(context.Request.Form["page"]),
  4. Rows = Convert.ToInt32(context.Request.Form["rows"]),
  5. Sort = context.Request.Form["sort"],
  6. Order = context.Request.Form["order"],
  7. };

然后,我们要访问数据:

  1. using (SchoolDatabase db = new SchoolDatabase()) {
  2. UserBO userBO = UserBO.CreateBO<UserBO>();
  3. IQueryable<UserBO> query = userBO.GetUserList(db);
  4. JArray jRows = new JArray();
  5. json.Add(new JProperty("total", query.Count()));
  6. //这里进行排序和分页操作
  7. query = datagridPara.ExecutePaging(query);
  8. foreach (UserBO bo in query) {
  9. JObject o = JObject.FromObject(bo);
  10. jRows.Add(JObject.FromObject(bo));
  11. }
  12. json.Add(new JProperty("rows", jRows));
  13. }

为了实现Queryable的排序和分页,我在DatagridParameters中增加了一个ExecutePaging方法,该方法用于将排序的字符串转换为Lambda排序,并添加分页功能。

  1. public class DatagridParameters {
  2. public int Page { get; set; }
  3. public int Rows { get; set; }
  4. public string Sort { get; set; }
  5. public string Order { get; set; }
  6. /// <summary>
  7. /// 执行分页操作
  8. /// </summary>
  9. /// <typeparam name="T"></typeparam>
  10. /// <param name="source"></param>
  11. /// <returns></returns>
  12. public IQueryable<T> ExecutePaging<T>(IQueryable<T> source) {
  13. if (string.IsNullOrWhiteSpace(Sort)) return source;
  14. if (string.IsNullOrWhiteSpace(Order)) Order = "asc";
  15. try {
  16. Type type = typeof(T);
  17. string methodName = "asc".Equals(Order, StringComparison.OrdinalIgnoreCase) ? "OrderBy" : "OrderByDescending";
  18. PropertyInfo property = type.GetProperty(Sort);
  19. ParameterExpression parameter = Expression.Parameter(type, "c");
  20. MemberExpression propertyAccess = Expression.MakeMemberAccess(parameter, property);
  21. LambdaExpression orderByExp = Expression.Lambda(propertyAccess, parameter);
  22. MethodCallExpression resultExp = Expression.Call(typeof(Queryable), methodName, new Type[] { type, property.PropertyType }, source.Expression, Expression.Quote(orderByExp));
  23. source = source.Provider.CreateQuery<T>(resultExp);
  24. } catch {}
  25. return source.Skip((Page - 1) * Rows).Take(Rows);
  26. }
  27. }