| /******************************************************************************* |
| * 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.Collections.Generic; |
| using System.Linq; |
| |
| namespace AppsheetEpplus; |
| |
| public class ExpressionCompiler : IExpressionCompiler { |
| private IEnumerable<Expression> _expressions; |
| private readonly IExpressionConverter _expressionConverter; |
| private readonly 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 []; |
| } |
| 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; |
| } |
| } |