| 
									
										
										
										
											2016-08-03 10:59:49 +00:00
										 |  |  |  | using System.Linq; | 
					
						
							|  |  |  |  | using System.Linq.Expressions; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace System.Web.UI.WebControls { | 
					
						
							|  |  |  |  |     public static class QueryExtensions { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         private const string SORT_DIRECTION_DESC = " DESC"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         public static IQueryable<T> SortBy<T>(this IQueryable<T> source, string sortExpression) where T : class { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (source == null) { | 
					
						
							|  |  |  |  |                 throw new ArgumentNullException("source"); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (String.IsNullOrWhiteSpace(sortExpression)) { | 
					
						
							|  |  |  |  |                 return source; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             sortExpression = sortExpression.Trim(); | 
					
						
							|  |  |  |  |             bool isDescending = false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // DataSource control passes the sort parameter with a direction | 
					
						
							|  |  |  |  |             // if the direction is descending | 
					
						
							|  |  |  |  |             if (sortExpression.EndsWith(SORT_DIRECTION_DESC, StringComparison.OrdinalIgnoreCase)) { | 
					
						
							|  |  |  |  |                 isDescending = true; | 
					
						
							|  |  |  |  |                 int descIndex = sortExpression.Length - SORT_DIRECTION_DESC.Length; | 
					
						
							|  |  |  |  |                 sortExpression = sortExpression.Substring(0, descIndex).Trim(); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (String.IsNullOrEmpty(sortExpression)) { | 
					
						
							|  |  |  |  |                 return source; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             ParameterExpression parameter = Expression.Parameter(source.ElementType, String.Empty); | 
					
						
							| 
									
										
										
										
											2016-11-10 13:04:39 +00:00
										 |  |  |  |             //VSO bug 173528-- Add support for sorting by nested property names | 
					
						
							|  |  |  |  |             MemberExpression property = null; | 
					
						
							|  |  |  |  |             string[] sortExpressionFields = sortExpression.Split('.'); | 
					
						
							|  |  |  |  |             foreach (string sortExpressionField in sortExpressionFields) { | 
					
						
							|  |  |  |  |                 if (property == null) { | 
					
						
							|  |  |  |  |                     property = Expression.Property(parameter, sortExpressionField); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |                 else { | 
					
						
							|  |  |  |  |                     property = Expression.Property(property, sortExpressionField); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-08-03 10:59:49 +00:00
										 |  |  |  |             LambdaExpression lambda = Expression.Lambda(property, parameter); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             string methodName = (isDescending) ? "OrderByDescending" : "OrderBy" ; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             Expression methodCallExpression = Expression.Call(typeof(Queryable), methodName, | 
					
						
							|  |  |  |  |                                                 new Type[] { source.ElementType, property.Type }, | 
					
						
							|  |  |  |  |                                                 source.Expression, Expression.Quote(lambda)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             return (IQueryable<T>)source.Provider.CreateQuery(methodCallExpression); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } |