dotnet-dynamic-linq/PoweredSoft.DynamicLinq/Fluent/WhereBuilder/WhereBuilder.cs
2018-03-14 18:26:30 -05:00

93 lines
2.8 KiB
C#

using PoweredSoft.DynamicLinq.Helpers;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Text;
using System.Threading.Tasks;
namespace PoweredSoft.DynamicLinq.Fluent
{
public class WhereBuilder<T> : WhereBuilderBase
{
public IQueryable<T> Query { get; set; }
public WhereBuilder(IQueryable<T> query)
{
Query = query;
}
public virtual IQueryable<T> Build()
{
// the query.
var query = Query;
if (Filters == null || Filters?.Count() == 0)
return query;
// shared parameter.
var sharedParameter = Expression.Parameter(typeof(T), "t");
// build the expression.
var filterExpressionMerged = BuildFilterExpression(sharedParameter, Filters);
// make changes on the query.
query = query.Where(filterExpressionMerged);
return query;
}
protected virtual Expression<Func<T, bool>> BuildFilterExpression(ParameterExpression parameter, List<WhereBuilderCondition> filters)
{
Expression<Func<T, bool>> temp = null;
filters.ForEach(filter =>
{
Expression<Func<T, bool>> innerExpression;
if (filter.Conditions?.Any() == true)
innerExpression = BuildFilterExpression(parameter, filter.Conditions);
else
innerExpression = BuildFilterExpression(parameter, filter);
if (temp == null)
{
temp = innerExpression;
}
else
{
if (filter.And)
temp = Expression.Lambda<Func<T, bool>>(Expression.AndAlso(temp.Body, innerExpression.Body), parameter);
else
temp = Expression.Lambda<Func<T, bool>>(Expression.OrElse(temp.Body, innerExpression.Body), parameter);
}
});
return temp;
}
protected virtual Expression<Func<T, bool>> BuildFilterExpression(ParameterExpression parameter, WhereBuilderCondition filter)
{
var ret = QueryableHelpers.CreateConditionExpression<T>(
filter.Path,
filter.ConditionOperator,
filter.Value,
filter.ConvertStrategy,
filter.CollectionHandling,
parameter: parameter,
nullChecking: IsNullCheckingEnabled,
stringComparision: filter.StringComparisation
);
return ret;
}
protected override WhereBuilderBase GetSubQueryBuilder()
{
return new WhereBuilder<T>(Query);
}
}
}