mirror of
				https://github.com/Biarity/Sieve.git
				synced 2025-10-25 14:56:26 +02:00 
			
		
		
		
	* Added option to disable creation of sort expression for nullable types * Changed property name. * Updated readme Co-authored-by: Teun Leijten <tleijten@inforit.nl>
		
			
				
	
	
		
			79 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			79 lines
		
	
	
		
			3.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Linq;
 | |
| using System.Linq.Expressions;
 | |
| using System.Reflection;
 | |
| 
 | |
| namespace Sieve.Extensions
 | |
| {
 | |
|     public static partial class LinqExtensions
 | |
|     {
 | |
|         public static IQueryable<TEntity> OrderByDynamic<TEntity>(
 | |
|             this IQueryable<TEntity> source,
 | |
|             string fullPropertyName,
 | |
|             PropertyInfo propertyInfo,
 | |
|             bool desc,
 | |
|             bool useThenBy, 
 | |
|             bool disableNullableTypeExpression = false)
 | |
|         {
 | |
|             var lambda = GenerateLambdaWithSafeMemberAccess<TEntity>(fullPropertyName, propertyInfo, disableNullableTypeExpression);
 | |
| 
 | |
|             var command = desc
 | |
|                 ? (useThenBy ? "ThenByDescending" : "OrderByDescending")
 | |
|                 : (useThenBy ? "ThenBy" : "OrderBy");
 | |
| 
 | |
|             var resultExpression = Expression.Call(
 | |
|                 typeof(Queryable),
 | |
|                 command,
 | |
|                 new Type[] { typeof(TEntity), lambda.ReturnType },
 | |
|                 source.Expression,
 | |
|                 Expression.Quote(lambda));
 | |
| 
 | |
|             return source.Provider.CreateQuery<TEntity>(resultExpression);
 | |
|         }
 | |
| 
 | |
|         private static Expression<Func<TEntity, object>> GenerateLambdaWithSafeMemberAccess<TEntity>
 | |
|         (
 | |
|             string fullPropertyName,
 | |
|             PropertyInfo propertyInfo,
 | |
|             bool disableNullableTypeExpression
 | |
|         )
 | |
|         {
 | |
|             var parameter = Expression.Parameter(typeof(TEntity), "e");
 | |
|             Expression propertyValue = parameter;
 | |
|             Expression nullCheck = null;
 | |
| 
 | |
|             foreach (var name in fullPropertyName.Split('.'))
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     propertyValue = Expression.PropertyOrField(propertyValue, name);
 | |
|                 }
 | |
|                 catch (ArgumentException)
 | |
|                 {
 | |
|                     // name is not a direct property of field of propertyValue expression. construct a memberAccess then.
 | |
|                     propertyValue = Expression.MakeMemberAccess(propertyValue, propertyInfo);
 | |
|                 }
 | |
| 
 | |
|                 if (propertyValue.Type.IsNullable() && !disableNullableTypeExpression)
 | |
|                 {
 | |
|                     nullCheck = GenerateOrderNullCheckExpression(propertyValue, nullCheck);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             var expression = nullCheck == null
 | |
|                 ? propertyValue
 | |
|                 : Expression.Condition(nullCheck, Expression.Default(propertyValue.Type), propertyValue);
 | |
| 
 | |
|             var converted = Expression.Convert(expression, typeof(object));
 | |
|             return Expression.Lambda<Func<TEntity, object>>(converted, parameter);
 | |
|         }
 | |
| 
 | |
|         private static Expression GenerateOrderNullCheckExpression(Expression propertyValue, Expression nullCheckExpression)
 | |
|         {
 | |
|             return nullCheckExpression == null
 | |
|                 ? Expression.Equal(propertyValue, Expression.Default(propertyValue.Type))
 | |
|                 : Expression.OrElse(nullCheckExpression, Expression.Equal(propertyValue, Expression.Default(propertyValue.Type)));
 | |
|         }
 | |
|     }
 | |
| }
 |