BiG*_*ERO 11 excel vba excel-2007 excel-vba
我在互联网上发现了很多资源,几乎可以做我想做的事情,但并不完全.我有一个命名范围"daylist".对于dayList中的每一天,我想在用户表单上创建一个按钮,该按钮将运行当天的宏.我可以动态添加按钮,但不知道如何将daycell.text从命名范围传递到按钮,事件处理程序,再到宏:S继承我必须创建用户表单的代码:
Sub addLabel()
ReadingsLauncher.Show vbModeless
Dim theLabel As Object
Dim labelCounter As Long
Dim daycell As Range
Dim btn As CommandButton
Dim btnCaption As String
For Each daycell In Range("daylist")
btnCaption = daycell.Text
Set theLabel = ReadingsLauncher.Controls.Add("Forms.Label.1", btnCaption, True)
With theLabel
.Caption = btnCaption
.Left = 10
.Width = 50
.Top = 20 * labelCounter
End With
Set btn = ReadingsLauncher.Controls.Add("Forms.CommandButton.1", "runButton", True)
With btn
.Caption = "Run Macro for " & btnCaption
.Left = 80
.Width = 80
.Top = 20 * labelCounter
' .OnAction = "btnPressed"
End With
labelCounter = labelCounter + 1
Next daycell
End Sub
Run Code Online (Sandbox Code Playgroud)
为了解决上述问题,我目前提示用户键入他们想要运行的日期(例如Day1)并将其传递给宏,它可以正常工作:
Sub B45runJoinTransactionAndFMMS()
loadDayNumber = InputBox("Please type the day you would like to load:", Title:="Enter Day", Default:="Day1")
Call JoinTransactionAndFMMS(loadDayNumber)
End Sub
Sub JoinTransactionAndFMMS(loadDayNumber As String)
xDayNumber = loadDayNumber
Sheets(xDayNumber).Activate
-Do stuff
End Sub
Run Code Online (Sandbox Code Playgroud)
因此,对于我的每个runButton,它需要显示daycell.text,并运行一个宏,该宏使用相同的文本作为参数来选择工作表来执行其中的操作.
任何帮助都是极好的.我已经看到动态写入vba代码的响应,以处理宏,但我相信必须有一些通过传递参数可以更优雅地完成,只是不确定如何.提前谢谢了!
SWa*_*SWa 19
我知道你现在已经接受了一个适合你的解决方案,并且比下面的方法简单得多,但是如果你感兴趣的话,这将是你问题的更直接的答案.
您需要创建一个类来处理按钮单击,因此每次单击该按钮时它都会使用该类中的事件,您只需要执行此操作一次,然后为每个按钮创建一个新实例.要阻止这些类超出范围并丢失,它们需要存储在类级别声明中.在下面我已经移动了你的代码一点点.
在类模块中(我称之为cButtonHandler)
Public WithEvents btn As MSForms.CommandButton
Private Sub btn_Click()
MsgBox btn.Caption
End Sub
Run Code Online (Sandbox Code Playgroud)
使用事件,因为它允许您使用大多数事件进行控制.我已将按钮生成代码移动到用户窗体中,如下所示:
Dim collBtns As Collection
Private Sub UserForm_Initialize()
Dim theLabel As Object
Dim labelCounter As Long
Dim daycell As Range
Dim btn As CommandButton
Dim btnCaption As String
'Create a variable of our events class
Dim btnH As cButtonHandler
'Create a new collection to hold the classes
Set collBtns = New Collection
For Each daycell In Range("daylist")
btnCaption = daycell.Text
Set theLabel = ReadingsLauncher.Controls.Add("Forms.Label.1", btnCaption, True)
With theLabel
.Caption = btnCaption
.Left = 10
.Width = 50
.Top = 20 * labelCounter
End With
Set btn = ReadingsLauncher.Controls.Add("Forms.CommandButton.1", "runButton", True)
With btn
.Caption = "Run Macro for " & btnCaption
.Left = 80
.Width = 80
.Top = 20 * labelCounter
'Create a new instance of our events class
Set btnH = New cButtonHandler
'Set the button we have created as the button in the class
Set btnH.btn = btn
'Add the class to the collection so it is not lost
'when this procedure finishes
collBtns.Add btnH
End With
labelCounter = labelCounter + 1
Next daycell
End Sub
Run Code Online (Sandbox Code Playgroud)
然后我们可以从一个单独的例程中调用useform:
Sub addLabel()
ReadingsLauncher.Show vbModeless
End Sub
Run Code Online (Sandbox Code Playgroud)
VBA中的课程并没有特别好地涵盖在许多VBA书籍中(通常你需要阅读VB6书籍以获得理解),但是一旦你理解了它们以及它们如何工作,它们就会变得非常有用:)
希望这可以帮助
编辑 - 解决其他查询
要引用集合中的对象,可以通过键或索引来完成.要使用该键,您需要在将项添加到集合时添加它,因此:
collBtns.Add btnH
Run Code Online (Sandbox Code Playgroud)
会成为
collBtns.Add btnH, btnCaption
Run Code Online (Sandbox Code Playgroud)
因此,密钥必须是唯一的.您可以参考如下:
'We refer to objects in a collection via the collection's key
'Or by it's place in the collection
'So either:
MsgBox collBtns("Monday").btn.Caption
'or:
MsgBox collBtns(1).btn.Caption
'We can then access it's properties and methods
'N.B you won't get any intellisense
collBtns("Monday").btn.Enabled = False
Run Code Online (Sandbox Code Playgroud)
如果需要,您还可以向类中添加其他属性/方法,例如:
Public WithEvents btn As MSForms.CommandButton
Private Sub btn_Click()
MsgBox btn.Caption
End Sub
Public Property Let Enabled(value As Boolean)
btn.Enabled = value
End Property
Run Code Online (Sandbox Code Playgroud)
然后将被访问:
collBtns("Monday").Enabled = False
Run Code Online (Sandbox Code Playgroud)
这有帮助吗?为了进一步阅读,我会指向Chip Pearson的网站,他在大多数主题上都有很棒的东西http://www.cpearson.com/excel/Events.aspx
请记住,VBA基于VB6,因此不是完全成熟的OO语言,例如,它不支持正常意义上的继承,只有接口继承
希望这可以帮助 :)