-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathArrayMap.cs
107 lines (94 loc) · 4.24 KB
/
ArrayMap.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
using System;
using ExcelDna.Integration;
using static ExcelDna.Integration.XlCall;
public static class ArrayMapFunctions
{
// This helper function is converted from https://github.com/Excel-DNA/Samples/blob/master/ArrayMap/Functions.vb
[ExcelFunction(Description = "Evaluates the two-argument function for every value in the first and second inputs. " + "Takes a single value and any rectangle, or one row and one column, or one column and one row.")]
public static object dnaArrayMap2([ExcelArgument(Description = "The function to evaluate - either enter the name without any quotes or brackets (for .xll functions), or as a string (for VBA functions)")] object function, [ExcelArgument(Description = "The input value(s) for the first argument (row, column or rectangular range) ")] object input1, [ExcelArgument(Description = "The input value(s) for the second argument (row, column or rectangular range) ")] object input2)
{
{
Func<object, object, object> evaluate;
if (function is double)
{
evaluate = (x, y) => Excel(xlUDF, function, x, y);
}
else if (function is string)
{
// First try to get the RegisterId, if it's an .xll UDF
object registerId;
registerId = Excel(xlfEvaluate, function);
if (registerId is double)
{
evaluate = (x, y) => Excel(xlUDF, registerId, x, y);
}
else
{
// Just call as string, hoping it's a valid VBA function
evaluate = (x, y) => Excel(xlUDF, function, x, y);
}
}
else
{
return ExcelError.ExcelErrorValue;
}
// Check for the case where one of the arguments is not an array, so we evaluate as a 1D function
if (!(input1 is object[,] inputArr1))
{
object evaluate1(object x) => evaluate(input1, x);
return ArrayEvaluate(evaluate1, input2);
}
if (!(input2 is object[,] inputArr2))
{
object evaluate1(object x) => evaluate(x, input2);
return ArrayEvaluate(evaluate1, input1);
}
// Otherwise we now have the function to evaluate, and two arrays
return ArrayEvaluate2(evaluate, inputArr1, inputArr2);
}
}
private static object[,] ArrayEvaluate2(Func<object, object, object> evaluate, object[,] inputArr1, object[,] inputArr2)
{
// Now we know both input1 and input2 are arrays
// We assume they are 1D, else we'll do our best to combine - the exact rules might be decided more carefully
if (inputArr1.GetLength(0) > 1)
{
// Lots of rows in input1, we'll take its first column only, and take the columns of input2
var rows = inputArr1.GetLength(0);
var cols = inputArr2.GetLength(1);
var output = new object[rows, cols];
for (int i = 0; i < rows; i++)
for (int j = 0; j < cols; j++)
output[i, j] = evaluate(inputArr1[i, 0], inputArr2[0, j]);
return output;
}
else
{
// Single row in input1, we'll take its columns, and take the rows from input2
var rows = inputArr2.GetLength(0);
var cols = inputArr1.GetLength(1);
var output = new object[rows, cols];
for (int i = 0; i < rows; i++)
for (int j = 0; j < cols; j++)
output[i, j] = evaluate(inputArr1[0, j], inputArr2[i, 0]);
return output;
}
}
private static object ArrayEvaluate(Func<object, object> evaluate, object input)
{
if (input is object[,] inputArr)
{
var rows = inputArr.GetLength(0);
var cols = inputArr.GetLength(1);
var output = new object[rows, cols];
for (int i = 0; i < rows; i++)
for (int j = 0; j < cols; j++)
output[i, j] = evaluate(inputArr[i, j]);
return output;
}
else
{
return evaluate(input);
}
}
}