Excel VBA CommandBar.OnAction with params is difficult / does not perform as expected

后端 未结 3 968
误落风尘
误落风尘 2021-01-19 01:09

So, I have Googled about and it seems that while making custom Pop up menus, if one wants to pass parameters then this is possible but for me comes with 2 major pro

相关标签:
3条回答
  • 2021-01-19 01:35

    You can use the .Parameter property. This is an example of a code in production (with only the lines of interest):

            Dim i As Integer
            Set cl = MainForm.Controls("classroomList")
            For i = 0 To cl.ListCount - 1
                With .Controls.Add(Type:=msoControlButton)
                    .Caption = cl.List(i)
                    .faceId = 177
                    .OnAction = "'" & ThisWorkbook.Name & "'!" & "assignClassroom"
                    .Parameter = cl.List(i)
                End With
            Next i
    

    And the procedure could be something like:

    Public Sub assignClassroom(Optional someArg as SomeType)
    ' code here
    CommandBars.ActionControl.Parameter 'The parameter here
    ' more code here
    End Sub
    
    0 讨论(0)
  • 2021-01-19 01:36

    The reason there are double calls and no break points is because of the parentheses (“( )”) surrounding the arguments in the .OnAction call:

        .OnAction = BuildProcArgString("MyProc", "A", "B", "C")
    

    Best guess: The parser for .OnAction chokes when these parentheses are used.

    This should work:

        .OnAction = "'BuildProcArgString" & chr(34) & "MyProc" & _
        chr(34) & "," & chr(34) & "A" & chr(34) & "," & chr(34) & _
        "B" & chr(34) & "," & chr(34) &  "C" &  "'"
    

    Other Notes:

    1) Single quotes, after the first double quote and before the last double quote, should be used to encapsulate the entire call.

    2) Chr(34) is the ASCII character for double quotes (“). All data types (ints, longs, strings, etc.), and quoted commas need to be preceeded by a Chr(34). The one exception is the ending sinlge quote (" ' "). Example:

        .OnAction = "'m_Test" & Chr(34) & 100 & Chr(34) & "," & Chr(34) & _
         intVariable & Chr(34) & "," & Chr(34) & "String" & Chr(34) & _
         "," & Chr(34) & stringVariable & "'"
    

    The function called:

        Public Function m_Test(i as Integer, iVar as Integer, s as String, sVar as String)
    

    3) .OnAction does not seem to pass Arrays or Objects. An item in an array can be passed (e.g. .OnAction = "'myTest" & Chr (34) & Args(0) & "'"), but not the entire Array (e.g. .OnAction = "'myTest" & Chr (34) & Args & "'"). Object pointers can be passed (ref: http://www.access-programmers.co.uk/forums/showthread.php?t=225415). But I've had no success in passing pointers to arrays.

    4) The .OnAction used in the original example is not surrounded by quotation marks so the .OnAction call is made when AssignIt() gets called but before the popup menu pops up.

    0 讨论(0)
  • 2021-01-19 01:46

    Don't ask me why this works, but it does. Source for this info is Using procedures with arguments in non-obvious instances

    Sub AssignIt()
    Const strCBarName As String = "MyNewPopupMenu"
    Dim cbrCmdBar As CommandBar
    
        'Delete it first so multiple runs can occur without appending
        On Error Resume Next
        Application.CommandBars(strCBarName).Delete
        On Error GoTo 0
    
        ' Create a pop-up menu.
        Set cbrCmdBar = Application.CommandBars.Add(Name:=strCBarName, Position:=msoBarPopup)
    
        'DEFINE COMMAND BAR CONTROL
        With Application.CommandBars(strCBarName).Controls.Add(Type:=msoControlButton)
            .Caption = "MyMenu"
            .OnAction = "'MyProc ""A"",""B"",2'"
        End With
        Application.CommandBars(strCBarName).ShowPopup
    End Sub
    
    Sub MyProc(x As String, y As String, z As Integer)
        MsgBox x & y & (z * 2)
        Debug.Print "AHA!!! the breakpoint works, and it's only called once!!!!!!"
    End Sub
    

    The key is to call the procedure in the .OnAction event surrounded by single quotes. Also, you need to escape your double quotes with double quotes. Numeric parameters need not be escaped.

    0 讨论(0)
提交回复
热议问题