vba - 多个类似的动态创建的命令按钮的操作

标签 vba excel dynamic commandbutton

我正在用 Excel VBA 编写一个程序,该程序基本上从一个文本框和一个命令按钮开始,命令按钮将在其下方创建一个新的文本框和命令按钮,而该命令按钮将依次创建一个新的文本框和命令按钮,等等。希望你跟着那个烂摊子。

我可以创建初始按钮没问题(它必须动态创建,以便以后有机会删除)。然后我的问题是创建 click()事件处理程序。我需要所有 click()事件做同样的事情,但命名新对象相对于它自己的名字。这一切都让我大吃一惊,我真的很感激一点帮助。

随意询问具体信息,但我还没有真正能够很好地围绕这个主题编写一些测试代码。

最佳答案

创建一个名为 CEventClass 的自定义类模块(插入 - 类模块,按 F4 更改名称)。将此代码键入到类模块中

'These are declared WithEvents so the events are
'exposed to us
Public WithEvents cmdEvent As MSForms.CommandButton
Public WithEvents tbxEvent As MSForms.TextBox


'This will fire for any control
'assigned to cmdEvent
Private Sub cmdEvent_Click()

    MsgBox cmdEvent.Caption

End Sub

'This will fire for any control
'assigned to tbxEvent
Private Sub tbxEvent_Change()

    If Len(tbxEvent.Text) < 6 Then
        tbxEvent.BackColor = vbYellow
    Else
        tbxEvent.BackColor = vbWhite
    End If

End Sub

现在创建一个没有控件的用户窗体。将此代码放入表单的代码模块中
'These will keep the class instances in
'scope for as long as the form is loaded
Private mEventButtons As Collection
Private mEventTexts As Collection

Private Sub UserForm_Initialize()

    Dim cmd As MSForms.CommandButton
    Dim txt As MSForms.TextBox
    Dim clsEventClass As CEventClass

    Set mEventButtons = New Collection
    Set mEventTexts = New Collection

    'Create two commandbuttons
    Set cmd = Me.Controls.Add("Forms.CommandButton.1", "FirstName")
    cmd.Top = 10
    cmd.Left = 10
    cmd.Caption = "First"
    'Create a new instance of CEventClass and
    'assign the button to cmdEvent
    Set clsEventClass = New CEventClass
    Set clsEventClass.cmdEvent = cmd
    mEventButtons.Add clsEventClass

    Set cmd = Me.Controls.Add("Forms.CommandButton.1", "SecondName")
    cmd.Top = 50
    cmd.Left = 10
    cmd.Caption = "Second"
    Set clsEventClass = New CEventClass
    Set clsEventClass.cmdEvent = cmd
    mEventButtons.Add clsEventClass

    'Create two textboxes and assign them to new instances
    'of the class
    Set txt = Me.Controls.Add("Forms.TextBox.1", "ThirdName")
    txt.Top = 10
    txt.Left = 150
    Set clsEventClass = New CEventClass
    Set clsEventClass.tbxEvent = txt
    mEventTexts.Add clsEventClass

    Set txt = Me.Controls.Add("Forms.TextBox.1", "FourthName")
    txt.Top = 50
    txt.Left = 150
    Set clsEventClass = New CEventClass
    Set clsEventClass.tbxEvent = txt
    mEventTexts.Add clsEventClass

End Sub

现在,当您运行表单时,如果您单击/更改控件,这两个事件将触发。

您可能会注意到文本框没有 AfterUpdate 事件。该事件实际上不是文本框事件,而是文本框的控件容器的事件,因此您不能以这种方式公开它。这就是我更喜欢在设计时创建所有控件并根据需要隐藏或取消隐藏它们的原因之一。我仍然可以将 WithEvents 用于某些控件,这样我就不必过多地重复代码。但是对于像 TextBox_AfterUpdate 这样的东西,我只是在设计时创建了所有的事件过程。

更新:

如果您希望事件创建新按钮,则必须做更多的事情。首先,您必须在用户窗体之外公开集合。您将其添加到您的用户表单模块
Public Property Get EventButtons() As Collection
    Set EventButtons = mEventButtons
End Property

然后你改变你的命令按钮事件代码来创建一个新按钮
Private Sub cmdEvent_Click()

    Dim cmd As MSForms.CommandButton
    Dim clsEventClass As CEventClass

    Set cmd = cmdEvent.Parent.Controls.Add("Forms.CommandButton.1", cmdEvent.Caption & "1")
    cmd.Top = cmdEvent.Top + 40
    cmd.Left = cmdEvent.Left
    cmd.Caption = cmdEvent.Caption & "1"
    Set clsEventClass = New CEventClass
    Set clsEventClass.cmdEvent = cmd
    cmdEvent.Parent.EventButtons.Add clsEventClass

End Sub

这将创建一个新按钮 40 点低于单击的任何一个。你没有说你的命名或定位逻辑是什么,所以我假设你可以解决这个问题。使用 cmdEvent.Parent 获取对 Userform 的引用。

关于vba - 多个类似的动态创建的命令按钮的操作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24558214/

相关文章:

excel - 如何删除前缀的动态数字

excel - 如何在 VBA 中设置和使用空范围?

excel - 如何在未安装 Excel 的情况下使用 Powershell 将 .xls 转换为 .csv

mfc - 在 MFC 中动态创建控件(集合问题)

MySQL/Mariadb 存储过程、准备语句、联合、动态创建的表和列名中的值

excel - 从字符串中提取特定长度的数字并用这些数字创建新字符串

sql - Access 查询中的参数?

python - 无法将 pandas Dataframe 附加到现有 Excel 工作表

excel - 数据透视表自定义聚合函数

c# - 如何处理动态表名