| /* 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-12-03 | 
 |  *******************************************************************************/ | 
 |  | 
 | using System; | 
 | using System.Collections.Generic; | 
 | using System.Linq; | 
 | using System.Text.RegularExpressions; | 
 | using OfficeOpenXml.FormulaParsing.Exceptions; | 
 | using OfficeOpenXml.FormulaParsing.ExpressionGraph; | 
 | using OfficeOpenXml.FormulaParsing.Utilities; | 
 |  | 
 | namespace OfficeOpenXml.FormulaParsing.Excel.Functions; | 
 |  | 
 | /// <summary> | 
 | /// Base class for Excel function implementations. | 
 | /// </summary> | 
 | public abstract class ExcelFunction { | 
 |   public ExcelFunction() | 
 |       : this(new(), new(), new()) {} | 
 |  | 
 |   public ExcelFunction( | 
 |       ArgumentCollectionUtil argumentCollectionUtil, | 
 |       ArgumentParsers argumentParsers, | 
 |       CompileResultValidators compileResultValidators) { | 
 |     _argumentCollectionUtil = argumentCollectionUtil; | 
 |     _argumentParsers = argumentParsers; | 
 |     _compileResultValidators = compileResultValidators; | 
 |   } | 
 |  | 
 |   private readonly ArgumentCollectionUtil _argumentCollectionUtil; | 
 |   private readonly ArgumentParsers _argumentParsers; | 
 |   private readonly CompileResultValidators _compileResultValidators; | 
 |  | 
 |   /// <summary> | 
 |   /// | 
 |   /// </summary> | 
 |   /// <param name="arguments">Arguments to the function, each argument can contain primitive types, lists or <see cref="ExcelDataProvider.IRangeInfo">Excel ranges</see></param> | 
 |   /// <param name="context">The <see cref="ParsingContext"/> contains various data that can be useful in functions.</param> | 
 |   /// <returns>A <see cref="CompileResult"/> containing the calculated value</returns> | 
 |   public abstract CompileResult Execute( | 
 |       IEnumerable<FunctionArgument> arguments, | 
 |       ParsingContext context); | 
 |  | 
 |   /// <summary> | 
 |   /// If overridden, this method is called before Execute is called. | 
 |   /// </summary> | 
 |   /// <param name="context"></param> | 
 |   public virtual void BeforeInvoke(ParsingContext context) {} | 
 |  | 
 |   public virtual bool IsLookupFuction => false; | 
 |  | 
 |   public virtual bool IsErrorHandlingFunction => false; | 
 |  | 
 |   /// <summary> | 
 |   /// Used for some Lookupfunctions to indicate that function arguments should | 
 |   /// not be compiled before the function is called. | 
 |   /// </summary> | 
 |   public bool SkipArgumentEvaluation { get; set; } | 
 |  | 
 |   protected object GetFirstValue(IEnumerable<FunctionArgument> val) { | 
 |     var arg = val.FirstOrDefault(); | 
 |     if (arg.Value is ExcelDataProvider.IRangeInfo) { | 
 |       //var r=((ExcelDataProvider.IRangeInfo)arg); | 
 |       var r = arg.ValueAsRangeInfo; | 
 |       return r.GetValue(r.Address._fromRow, r.Address._fromCol); | 
 |     } | 
 |     return arg?.Value; | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// This functions validates that the supplied <paramref name="arguments"/> contains at least | 
 |   /// (the value of) <paramref name="minLength"/> elements. If one of the arguments is an | 
 |   /// <see cref="ExcelDataProvider.IRangeInfo">Excel range</see> the number of cells in | 
 |   /// that range will be counted as well. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="minLength"></param> | 
 |   /// <param name="errorTypeToThrow">The <see cref="eErrorType"/> of the <see cref="ExcelErrorValueException"/> that will be thrown if <paramref name="minLength"/> is not met.</param> | 
 |   protected void ValidateArguments( | 
 |       IEnumerable<FunctionArgument> arguments, | 
 |       int minLength, | 
 |       eErrorType errorTypeToThrow) { | 
 |     Require.That(arguments).Named("arguments").IsNotNull(); | 
 |     ThrowExcelErrorValueExceptionIf( | 
 |         () => { | 
 |           var nArgs = 0; | 
 |           if (arguments.Any()) { | 
 |             foreach (var arg in arguments) { | 
 |               nArgs++; | 
 |               if (nArgs >= minLength) { | 
 |                 return false; | 
 |               } | 
 |               if (arg.IsExcelRange) { | 
 |                 nArgs += arg.ValueAsRangeInfo.GetNCells(); | 
 |                 if (nArgs >= minLength) { | 
 |                   return false; | 
 |                 } | 
 |               } | 
 |             } | 
 |           } | 
 |           return true; | 
 |         }, | 
 |         errorTypeToThrow); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// This functions validates that the supplied <paramref name="arguments"/> contains at least | 
 |   /// (the value of) <paramref name="minLength"/> elements. If one of the arguments is an | 
 |   /// <see cref="ExcelDataProvider.IRangeInfo">Excel range</see> the number of cells in | 
 |   /// that range will be counted as well. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="minLength"></param> | 
 |   /// <exception cref="ArgumentException"></exception> | 
 |   protected void ValidateArguments(IEnumerable<FunctionArgument> arguments, int minLength) { | 
 |     Require.That(arguments).Named("arguments").IsNotNull(); | 
 |     ThrowArgumentExceptionIf( | 
 |         () => { | 
 |           var nArgs = 0; | 
 |           if (arguments.Any()) { | 
 |             foreach (var arg in arguments) { | 
 |               nArgs++; | 
 |               if (nArgs >= minLength) { | 
 |                 return false; | 
 |               } | 
 |               if (arg.IsExcelRange) { | 
 |                 nArgs += arg.ValueAsRangeInfo.GetNCells(); | 
 |                 if (nArgs >= minLength) { | 
 |                   return false; | 
 |                 } | 
 |               } | 
 |             } | 
 |           } | 
 |           return true; | 
 |         }, | 
 |         "Expecting at least {0} arguments", | 
 |         minLength.ToString()); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Returns the value of the argument att the position of the 0-based | 
 |   /// <paramref name="index"/> as an integer. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="index"></param> | 
 |   /// <returns>Value of the argument as an integer.</returns> | 
 |   /// <exception cref="ExcelErrorValueException"></exception> | 
 |   protected int ArgToInt(IEnumerable<FunctionArgument> arguments, int index) { | 
 |     var val = arguments.ElementAt(index).ValueFirst; | 
 |     return (int)_argumentParsers.GetParser(DataType.Integer).Parse(val); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Returns the value of the argument att the position of the 0-based | 
 |   /// <paramref name="index"/> as a string. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="index"></param> | 
 |   /// <returns>Value of the argument as a string.</returns> | 
 |   protected string ArgToString(IEnumerable<FunctionArgument> arguments, int index) { | 
 |     var obj = arguments.ElementAt(index).ValueFirst; | 
 |     return obj != null ? obj.ToString() : string.Empty; | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Returns the value of the argument att the position of the 0-based | 
 |   /// </summary> | 
 |   /// <param name="obj"></param> | 
 |   /// <returns>Value of the argument as a double.</returns> | 
 |   /// <exception cref="ExcelErrorValueException"></exception> | 
 |   protected double ArgToDecimal(object obj) { | 
 |     return (double)_argumentParsers.GetParser(DataType.Decimal).Parse(obj); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Returns the value of the argument att the position of the 0-based | 
 |   /// <paramref name="index"/> as a <see cref="System.Double"/>. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="index"></param> | 
 |   /// <returns>Value of the argument as an integer.</returns> | 
 |   /// <exception cref="ExcelErrorValueException"></exception> | 
 |   protected double ArgToDecimal(IEnumerable<FunctionArgument> arguments, int index) { | 
 |     return ArgToDecimal(arguments.ElementAt(index).Value); | 
 |   } | 
 |  | 
 |   protected double Divide(double left, double right) { | 
 |     if (System.Math.Abs(right - 0d) < double.Epsilon) { | 
 |       throw new ExcelErrorValueException(eErrorType.Div0); | 
 |     } | 
 |     return left / right; | 
 |   } | 
 |  | 
 |   protected bool IsNumericString(object value) { | 
 |     if (value == null || string.IsNullOrEmpty(value.ToString())) { | 
 |       return false; | 
 |     } | 
 |     return Regex.IsMatch(value.ToString(), @"^[\d]+(\,[\d])?"); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// If the argument is a boolean value its value will be returned. | 
 |   /// If the argument is an integer value, true will be returned if its | 
 |   /// value is not 0, otherwise false. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="index"></param> | 
 |   /// <returns></returns> | 
 |   protected bool ArgToBool(IEnumerable<FunctionArgument> arguments, int index) { | 
 |     var obj = arguments.ElementAt(index).Value ?? string.Empty; | 
 |     return (bool)_argumentParsers.GetParser(DataType.Boolean).Parse(obj); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Throws an <see cref="ArgumentException"/> if <paramref name="condition"/> evaluates to true. | 
 |   /// </summary> | 
 |   /// <param name="condition"></param> | 
 |   /// <param name="message"></param> | 
 |   /// <exception cref="ArgumentException"></exception> | 
 |   protected void ThrowArgumentExceptionIf(Func<bool> condition, string message) { | 
 |     if (condition()) { | 
 |       throw new ArgumentException(message); | 
 |     } | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Throws an <see cref="ArgumentException"/> if <paramref name="condition"/> evaluates to true. | 
 |   /// </summary> | 
 |   /// <param name="condition"></param> | 
 |   /// <param name="message"></param> | 
 |   /// <param name="formats">Formats to the message string.</param> | 
 |   protected void ThrowArgumentExceptionIf( | 
 |       Func<bool> condition, | 
 |       string message, | 
 |       params object[] formats) { | 
 |     message = string.Format(message, formats); | 
 |     ThrowArgumentExceptionIf(condition, message); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Throws an <see cref="ExcelErrorValueException"/> with the given <paramref name="errorType"/> set. | 
 |   /// </summary> | 
 |   /// <param name="errorType"></param> | 
 |   protected void ThrowExcelErrorValueException(eErrorType errorType) { | 
 |     throw new ExcelErrorValueException( | 
 |         "An excel function error occurred", | 
 |         ExcelErrorValue.Create(errorType)); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Throws an <see cref="ArgumentException"/> if <paramref name="condition"/> evaluates to true. | 
 |   /// </summary> | 
 |   /// <param name="condition"></param> | 
 |   /// <param name="errorType"></param> | 
 |   /// <exception cref="ExcelErrorValueException"></exception> | 
 |   protected void ThrowExcelErrorValueExceptionIf(Func<bool> condition, eErrorType errorType) { | 
 |     if (condition()) { | 
 |       throw new ExcelErrorValueException( | 
 |           "An excel function error occurred", | 
 |           ExcelErrorValue.Create(errorType)); | 
 |     } | 
 |   } | 
 |  | 
 |   protected bool IsNumeric(object val) { | 
 |     if (val == null) { | 
 |       return false; | 
 |     } | 
 |     return (val.GetType().IsPrimitive | 
 |             || val is double | 
 |             || val is decimal | 
 |             || val is System.DateTime | 
 |             || val is TimeSpan); | 
 |   } | 
 |  | 
 |   //protected virtual bool IsNumber(object obj) | 
 |   //{ | 
 |   //    if (obj == null) return false; | 
 |   //    return (obj is int || obj is double || obj is short || obj is decimal || obj is long); | 
 |   //} | 
 |  | 
 |   /// <summary> | 
 |   /// Helper method for comparison of two doubles. | 
 |   /// </summary> | 
 |   /// <param name="d1"></param> | 
 |   /// <param name="d2"></param> | 
 |   /// <returns></returns> | 
 |   protected bool AreEqual(double d1, double d2) { | 
 |     return System.Math.Abs(d1 - d2) < double.Epsilon; | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Will return the arguments as an enumerable of doubles. | 
 |   /// </summary> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="context"></param> | 
 |   /// <returns></returns> | 
 |   protected virtual IEnumerable<double> ArgsToDoubleEnumerable( | 
 |       IEnumerable<FunctionArgument> arguments, | 
 |       ParsingContext context) { | 
 |     return ArgsToDoubleEnumerable(false, arguments, context); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Will return the arguments as an enumerable of doubles. | 
 |   /// </summary> | 
 |   /// <param name="ignoreHiddenCells">If a cell is hidden and this value is true the value of that cell will be ignored</param> | 
 |   /// <param name="ignoreErrors">If a cell contains an error, that error will be ignored if this method is set to true</param> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="context"></param> | 
 |   /// <returns></returns> | 
 |   protected virtual IEnumerable<double> ArgsToDoubleEnumerable( | 
 |       bool ignoreHiddenCells, | 
 |       bool ignoreErrors, | 
 |       IEnumerable<FunctionArgument> arguments, | 
 |       ParsingContext context) { | 
 |     return _argumentCollectionUtil.ArgsToDoubleEnumerable( | 
 |         ignoreHiddenCells, | 
 |         ignoreErrors, | 
 |         arguments, | 
 |         context); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Will return the arguments as an enumerable of doubles. | 
 |   /// </summary> | 
 |   /// <param name="ignoreHiddenCells">If a cell is hidden and this value is true the value of that cell will be ignored</param> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="context"></param> | 
 |   /// <returns></returns> | 
 |   protected virtual IEnumerable<double> ArgsToDoubleEnumerable( | 
 |       bool ignoreHiddenCells, | 
 |       IEnumerable<FunctionArgument> arguments, | 
 |       ParsingContext context) { | 
 |     return ArgsToDoubleEnumerable(ignoreHiddenCells, true, arguments, context); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Will return the arguments as an enumerable of objects. | 
 |   /// </summary> | 
 |   /// <param name="ignoreHiddenCells">If a cell is hidden and this value is true the value of that cell will be ignored</param> | 
 |   /// <param name="arguments"></param> | 
 |   /// <param name="context"></param> | 
 |   /// <returns></returns> | 
 |   protected virtual IEnumerable<object> ArgsToObjectEnumerable( | 
 |       bool ignoreHiddenCells, | 
 |       IEnumerable<FunctionArgument> arguments, | 
 |       ParsingContext context) { | 
 |     return _argumentCollectionUtil.ArgsToObjectEnumerable(ignoreHiddenCells, arguments, context); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Use this method to create a result to return from Excel functions. | 
 |   /// </summary> | 
 |   /// <param name="result"></param> | 
 |   /// <param name="dataType"></param> | 
 |   /// <returns></returns> | 
 |   protected CompileResult CreateResult(object result, DataType dataType) { | 
 |     var validator = _compileResultValidators.GetValidator(dataType); | 
 |     validator.Validate(result); | 
 |     return new(result, dataType); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// Use this method to apply a function on a collection of arguments. The <paramref name="result"/> | 
 |   /// should be modifyed in the supplied <paramref name="action"/> and will contain the result | 
 |   /// after this operation has been performed. | 
 |   /// </summary> | 
 |   /// <param name="collection"></param> | 
 |   /// <param name="result"></param> | 
 |   /// <param name="action"></param> | 
 |   /// <returns></returns> | 
 |   protected virtual double CalculateCollection( | 
 |       IEnumerable<FunctionArgument> collection, | 
 |       double result, | 
 |       Func<FunctionArgument, double, double> action) { | 
 |     return _argumentCollectionUtil.CalculateCollection(collection, result, action); | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// if the supplied <paramref name="arg">argument</paramref> contains an Excel error | 
 |   /// an <see cref="ExcelErrorValueException"/> with that errorcode will be thrown | 
 |   /// </summary> | 
 |   /// <param name="arg"></param> | 
 |   /// <exception cref="ExcelErrorValueException"></exception> | 
 |   protected void CheckForAndHandleExcelError(FunctionArgument arg) { | 
 |     if (arg.ValueIsExcelError) { | 
 |       throw (new ExcelErrorValueException(arg.ValueAsExcelErrorValue)); | 
 |     } | 
 |   } | 
 |  | 
 |   /// <summary> | 
 |   /// If the supplied <paramref name="cell"/> contains an Excel error | 
 |   /// an <see cref="ExcelErrorValueException"/> with that errorcode will be thrown | 
 |   /// </summary> | 
 |   /// <param name="cell"></param> | 
 |   protected void CheckForAndHandleExcelError(ExcelDataProvider.ICellInfo cell) { | 
 |     if (cell.IsExcelError) { | 
 |       throw (new ExcelErrorValueException(ExcelErrorValue.Parse(cell.Value.ToString()))); | 
 |     } | 
 |   } | 
 |  | 
 |   protected CompileResult GetResultByObject(object result) { | 
 |     if (IsNumeric(result)) { | 
 |       return CreateResult(result, DataType.Decimal); | 
 |     } | 
 |     if (result is string) { | 
 |       return CreateResult(result, DataType.String); | 
 |     } | 
 |     if (ExcelErrorValue.Values.IsErrorValue(result)) { | 
 |       return CreateResult(result, DataType.ExcelAddress); | 
 |     } | 
 |     if (result == null) { | 
 |       return CompileResult.Empty; | 
 |     } | 
 |     return CreateResult(result, DataType.Enumerable); | 
 |   } | 
 | } |