UDF Excel that accepts both Range and Array as Parameter

Kanzisa

New Member
Joined
Feb 6, 2024
Messages
2
Office Version
  1. 2021
Platform
  1. Windows
I am trying to create a new UDF MaxDrawdown_General_Purpose that takes the values from selected cells or range of cells and at the same time that takes the value from an array in VBA ambient. How can i merge 2 functions below in a new one MaxDrawdown_General_Purpose ? Thank you in advance.

VBA Code:
Function MaxDrawdown(data() As Double) As Variant 'Runs only in VBA ambient
         Dim N As Integer
         Dim i As Integer
         Dim highestPrice As Double
         Dim currentPrice As Double
         Dim currentDrawdown As Double
            
         N = UBound(data)
        
         If N > 0 Then
             highestPrice = 1
             MaxDrawdown = 0
    
             For i = 0 To N - 1
                 currentPrice = highestPrice * (1 + data(i) / 100)
                 currentDrawdown = (highestPrice - currentPrice) /      highestPrice
                 highestPrice = IIf(currentPrice > highestPrice,      currentPrice, highestPrice)
                 MaxDrawdown = IIf(currentDrawdown > MaxDrawdown,      currentDrawdown, MaxDrawdown)
             Next i
    
             MaxDrawdown = 100 * MaxDrawdown
         Else
             MaxDrawdown = "undefined"
         End If
     End Function
    
    
    
Function MaxDrawdownExcel2(rng As Range) As Double  'Runs only in Excel
        
         Dim N As Integer
         Dim i As Integer
         Dim highestPrice As Double
         Dim currentPrice As Double
         Dim currentDrawdown As Double
         Dim data() As Variant
        
         data = rng.Value
     
         N = UBound(data)
    
         If N > 0 Then
             highestPrice = 1
             MaxDrawdownExcel2 = 0
    
             For i = 1 To N
                 currentPrice = highestPrice * (1 + data(i, 1) / 100)
                 currentDrawdown = (highestPrice - currentPrice) /      highestPrice
                 highestPrice = IIf(currentPrice > highestPrice,      currentPrice, highestPrice)
                 MaxDrawdownExcel2 = IIf(currentDrawdown >      MaxDrawdownExcel2, currentDrawdown, MaxDrawdownExcel2)
             Next i
    
             MaxDrawdownExcel2 = 100 * MaxDrawdownExcel2
         Else
             MaxDrawdownExcel2 = "undefined"
         End If
     End Function
 
Last edited by a moderator:

Excel Facts

Copy formula down without changing references
If you have =SUM(F2:F49) in F50; type Alt+' in F51 to copy =SUM(F2:F49) to F51, leaving the formula in edit mode. Change SUM to COUNT.
First, declare data as Variant.

Then, check whether data is a Range object. If so, convert it to an array.

Then, instead of using index, use For Each/Next to loop through each item in the array. The reason for this is that data may contain either a one dimensional or two dimensional array. And so the For Each/Next loop easily handles both these situations.

VBA Code:
Function MaxDrawdown_General_Purpose(data As Variant) As Variant

    Dim highestPrice As Double
    Dim MaxDrawdown As Double
    Dim itm As Variant

    If TypeOf data Is Range Then
        data = data
    End If
    
    highestPrice = 1
    MaxDrawdown = 0

    For Each itm In data
        currentPrice = highestPrice * (1 + itm / 100)
        currentDrawdown = (highestPrice - currentPrice) / highestPrice
        highestPrice = IIf(currentPrice > highestPrice, currentPrice, highestPrice)
        MaxDrawdown = IIf(currentDrawdown > MaxDrawdown, currentDrawdown, MaxDrawdown)
    Next itm

    MaxDrawdown = 100 * MaxDrawdown
    
End Function

Hope this helps!
 
Upvote 0
Thank you for your suggestions.
I run your code (with some integration data=data.Value) but the output is ""

Function MaxDrawdown_General_Purpose(data As Variant) As Variant

Dim highestPrice As Double
Dim MaxDrawdown As Double
Dim itm As Variant
Dim currentPrice As Double

Dim currentDrawdown As Double

If TypeOf data Is Range Then
data = data.Value
End If

highestPrice = 1
MaxDrawdown = 0

For Each itm In data
currentPrice = highestPrice * (1 + itm / 100)
currentDrawdown = (highestPrice - currentPrice) / highestPrice
highestPrice = IIf(currentPrice > highestPrice, currentPrice, highestPrice)
MaxDrawdown = IIf(currentDrawdown > MaxDrawdown, currentDrawdown, MaxDrawdown)
Next itm

MaxDrawdown = 100 * MaxDrawdown

End Function

I seems to me that using the test "If TypeOf data Is Range might not work as expected if data is an array. To me, we should check the data type differently

If TypeName(data) = "Range" Then
data = data.Value
End If.

But I consider myself only a practitioner, i wouldn't know how to procede.
 
Upvote 0
There should be no need to make any changes to my code. It should remain as data = data
 
Upvote 0

Forum statistics

Threads
1,225,743
Messages
6,186,770
Members
453,370
Latest member
juliewar

We've detected that you are using an adblocker.

We have a great community of people providing Excel help here, but the hosting costs are enormous. You can help keep this site running by allowing ads on MrExcel.com.
Allow Ads at MrExcel

Which adblocker are you using?

Disable AdBlock

Follow these easy steps to disable AdBlock

1)Click on the icon in the browser’s toolbar.
2)Click on the icon in the browser’s toolbar.
2)Click on the "Pause on this site" option.
Go back

Disable AdBlock Plus

Follow these easy steps to disable AdBlock Plus

1)Click on the icon in the browser’s toolbar.
2)Click on the toggle to disable it for "mrexcel.com".
Go back

Disable uBlock Origin

Follow these easy steps to disable uBlock Origin

1)Click on the icon in the browser’s toolbar.
2)Click on the "Power" button.
3)Click on the "Refresh" button.
Go back

Disable uBlock

Follow these easy steps to disable uBlock

1)Click on the icon in the browser’s toolbar.
2)Click on the "Power" button.
3)Click on the "Refresh" button.
Go back
Back
Top