| /******************************************************************************* | 
 |  * You may amend and distribute as you like, but don't remove this header! | 
 |  * | 
 |  * EPPlus provides server-side generation of Excel 2007/2010 spreadsheets. | 
 |  * See http://www.codeplex.com/EPPlus for details. | 
 |  * | 
 |  * Copyright (C) 2011  Jan Källman | 
 |  * | 
 |  * This library is free software; you can redistribute it and/or | 
 |  * modify it under the terms of the GNU Lesser General Public | 
 |  * License as published by the Free Software Foundation; either | 
 |  * version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |  * This library is distributed in the hope that it will be useful, | 
 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.   | 
 |  * See the GNU Lesser General Public License for more details. | 
 |  * | 
 |  * The GNU Lesser General Public License can be viewed at http://www.opensource.org/licenses/lgpl-license.php | 
 |  * If you unfamiliar with this license or have questions about it, here is an http://www.gnu.org/licenses/gpl-faq.html | 
 |  * | 
 |  * All code and executables are provided "as is" with no warranty either express or implied.  | 
 |  * The author accepts no liability for any damage or loss of business that this product may cause. | 
 |  * | 
 |  * Code change notes: | 
 |  *  | 
 |  * Author							Change						Date | 
 |  * ****************************************************************************** | 
 |  * Mats Alm   		                Added       		        2013-03-01 (Prior file history on https://github.com/swmal/ExcelFormulaParser) | 
 |  *******************************************************************************/ | 
 | using System; | 
 | using System.Collections.Generic; | 
 | using System.Linq; | 
 | using System.Text; | 
 | using OfficeOpenXml.FormulaParsing.Excel.Operators; | 
 | using OfficeOpenXml.FormulaParsing.ExpressionGraph.CompileStrategy; | 
 |  | 
 | namespace OfficeOpenXml.FormulaParsing.ExpressionGraph | 
 | { | 
 |     public class ExpressionCompiler : IExpressionCompiler | 
 |     { | 
 |         private IEnumerable<Expression> _expressions; | 
 |         private IExpressionConverter _expressionConverter; | 
 |         private ICompileStrategyFactory _compileStrategyFactory; | 
 |  | 
 |         public ExpressionCompiler() | 
 |             : this(new ExpressionConverter(), new CompileStrategyFactory()) | 
 |         { | 
 |   | 
 |         } | 
 |  | 
 |         public ExpressionCompiler(IExpressionConverter expressionConverter, ICompileStrategyFactory compileStrategyFactory) | 
 |         { | 
 |             _expressionConverter = expressionConverter; | 
 |             _compileStrategyFactory = compileStrategyFactory; | 
 |         } | 
 |  | 
 |         public CompileResult Compile(IEnumerable<Expression> expressions) | 
 |         { | 
 |             _expressions = expressions; | 
 |             return PerformCompilation(); | 
 |         } | 
 |         public CompileResult Compile(string worksheet, int row, int column, IEnumerable<Expression> expressions) | 
 |         { | 
 |             _expressions = expressions; | 
 |             return PerformCompilation(worksheet, row, column); | 
 |         } | 
 |  | 
 |         private CompileResult PerformCompilation(string worksheet="", int row=-1, int column=-1) | 
 |         { | 
 |             var compiledExpressions = HandleGroupedExpressions(); | 
 |             while(compiledExpressions.Any(x => x.Operator != null)) | 
 |             { | 
 |                 var prec = FindLowestPrecedence(); | 
 |                 compiledExpressions = HandlePrecedenceLevel(prec); | 
 |             } | 
 |             if (_expressions.Any()) | 
 |             { | 
 |                 return compiledExpressions.First().Compile(); | 
 |             } | 
 |             return CompileResult.Empty; | 
 |         } | 
 |  | 
 |         private IEnumerable<Expression> HandleGroupedExpressions() | 
 |         { | 
 |             if (!_expressions.Any()) return Enumerable.Empty<Expression>(); | 
 |             var first = _expressions.First(); | 
 |             var groupedExpressions = _expressions.Where(x => x.IsGroupedExpression); | 
 |             foreach(var groupedExpression in groupedExpressions) | 
 |             { | 
 |                 var result = groupedExpression.Compile(); | 
 |                 if (result == CompileResult.Empty) continue; | 
 |                 var newExp = _expressionConverter.FromCompileResult(result); | 
 |                 newExp.Operator = groupedExpression.Operator; | 
 |                 newExp.Prev = groupedExpression.Prev; | 
 |                 newExp.Next = groupedExpression.Next; | 
 |                 if (groupedExpression.Prev != null) | 
 |                 { | 
 |                     groupedExpression.Prev.Next = newExp; | 
 |                 } | 
 |                 if (groupedExpression.Next != null) | 
 |                 { | 
 |                     groupedExpression.Next.Prev = newExp; | 
 |                 } | 
 |                 if (groupedExpression == first) | 
 |                 { | 
 |                     first = newExp; | 
 |                 } | 
 |             } | 
 |             return RefreshList(first); | 
 |         } | 
 |  | 
 |         private IEnumerable<Expression> HandlePrecedenceLevel(int precedence) | 
 |         { | 
 |             var first = _expressions.First(); | 
 |             var expressionsToHandle = _expressions.Where(x => x.Operator != null && x.Operator.Precedence == precedence); | 
 |             var last = expressionsToHandle.Last(); | 
 |             var expression = expressionsToHandle.First(); | 
 |             do | 
 |             { | 
 |                 var strategy = _compileStrategyFactory.Create(expression); | 
 |                 var compiledExpression = strategy.Compile(); | 
 |                 if(compiledExpression is ExcelErrorExpression) | 
 |                 { | 
 |                     return RefreshList(compiledExpression); | 
 |                 } | 
 |                 if (expression == first) | 
 |                 { | 
 |                     first = compiledExpression; | 
 |                 } | 
 |  | 
 |                 expression = compiledExpression; | 
 |             } | 
 |             while (expression != null && expression.Operator != null && expression.Operator.Precedence == precedence); | 
 |             return RefreshList(first); | 
 |         } | 
 |  | 
 |         private int FindLowestPrecedence() | 
 |         { | 
 |             return _expressions.Where(x => x.Operator != null).Min(x => x.Operator.Precedence); | 
 |         } | 
 |  | 
 |         private IEnumerable<Expression> RefreshList(Expression first) | 
 |         { | 
 |             var resultList = new List<Expression>(); | 
 |             var exp = first; | 
 |             resultList.Add(exp); | 
 |             while (exp.Next != null) | 
 |             { | 
 |                 resultList.Add(exp.Next); | 
 |                 exp = exp.Next; | 
 |             } | 
 |             _expressions = resultList; | 
 |             return resultList; | 
 |         } | 
 |     } | 
 | } |