- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我打开了一个插件和一个工作簿。该插件是一个 .xlam 文件,在工作簿中我添加了对它的引用。该插件受密码保护。
可以从我的工作簿中运行插件的公共(public)方法。但是,插件中的一种方法利用 VBA.UserForms.Add
打开在运行时创建的用户窗体 like this
假设包含对 myAddin
引用的工作簿具有以下内容:
Private Sub callAddin()
myAddin.ShowForm ThisWorkbook
End Sub
通常,我的插件中的代码如下所示:
Public Sub ShowForm(CallerWorkbook As Workbook)
Const vbext_ct_MSForm As Long = 3
'This is to stop screen flashing while creating form
Application.VBE.MainWindow.Visible = False
'Add to ThisWorkbook, not supplied workbook or VBE will crash - ignore CallerWorkbook
Dim myForm As Object
Set myForm = ThisWorkbook.VBProject.VBComponents.Add(vbext_ct_MSForm)
'Create the User Form
With myForm
.Properties("Caption") = "Select"
.Properties("Width") = 300
.Properties("Height") = 270
End With
'Show the form
Dim finalForm As Object
Set finalForm = VBA.UserForms.Add(myForm.Name)
finalForm.Show
'Remove form
ThisWorkbook.VBProject.VBComponents.Remove myForm
End Sub
效果很好。但是,当我的插件受密码保护时,不允许尝试向其中添加临时用户表单。没问题,我只需将临时用户窗体添加到调用代码的工作簿中,因为这不会受到密码保护
Sub ShowForm(CallerWorkbook As Workbook)
Const vbext_ct_MSForm As Long = 3
'This is to stop screen flashing while creating form
Application.VBE.MainWindow.Visible = False
'Add to CallerWorkbook instead
Dim myForm As Object
Set myForm = CallerWorkbook.VBProject.VBComponents.Add(vbext_ct_MSForm)
'Create the User Form
With myForm
.Properties("Caption") = "Select"
.Properties("Width") = 300
.Properties("Height") = 270
End With
'Show the form
Dim finalForm As Object
'Now myForm cannot be found and added
Set finalForm = VBA.UserForms.Add(myForm.Name)
finalForm.Show
'Remove form
CallerWorkbook.VBProject.VBComponents.Remove myForm
End Sub
但是,VBA 似乎无法看到 myForm.Name
现在指向的位置,因此 Add 方法失败,并显示“运行时错误 424:需要对象” “
有没有办法在另一个工作簿中显示运行时创建的表单?
最佳答案
您遇到的问题是用户窗体默认是私有(private)实例化的。这意味着一个项目无法引用另一个项目中的 UserForm,并且如果您无法引用该表单,则无法调用它的 Show
方法。
您的 Set myForm = CallerWorkbook.VBProject.VBComponents.Add(vbext_ct_MSForm)
语句返回 VbComponent
,而不是 UserForm
,这就是为什么然后您不能使用 VBA.UserForms.Add(myForm.Name)
有两种方法可以解决这个问题:
PublicNotCreatable
模板 UserFormUserForm 就像一个类,因此它可以像类一样设置其Instancing
属性。但是,VBE 不会在用户窗体的属性窗口中公开 Instancing
属性,因此要设置实例,您需要导出表单,然后编辑 Attribute VB_Expose
> 在再次导入表单之前,在文本编辑器中编辑 FRM 文件中的属性。步骤如下:
TemplateForm
的用户表单TemplateForm
并选择在删除之前导出表单TemplateForm.frm
文件Attribute VB_Expose = False
,使其读取为Attribute VB_Exposed = True
TemplateForm.frm
TemplateForm.frm
导入您的加载项添加一个公共(public)函数,该函数将 TemplateForm
的新实例返回到您的外接程序。我已使此函数接受工作簿引用,以便加载项可以配置表单上的任何工作簿特定属性:
Public Function GetTemplateForm(CallerWorkbook As Workbook) As TemplateForm
Dim frm As TemplateForm
Set frm = New TemplateForm
'Set early-bound properties with intellisense
frm.Caption = "Select"
frm.Width = 300
frm.Height = 270
'Configure CallerWorkbook specific form properties here
'...
Set GetTemplateForm = frm
End Function
然后,您可以在用户的工作簿中显示 TemplateForm 的实例,而无需动态添加表单,或处理屏幕闪烁或难以调试的代码:
Sub ShowAddinForm()
With MyAddin.GetTemplateForm(ThisWorkbook)
'Do more workbook specific propery setting here...
'...
.Show
End With
End Sub
** 注意 - Rubberduck VBA 插件很快就能添加 PublicNotCreatable
用户窗体。
这种方法不太优雅。用户需要管理更多代码,并且屏幕闪烁,并且难以调试代码。步骤如下:
将此代码添加到加载项:
Public Function GetTempFormName(CallerWorkbook As Workbook) As String
Const vbext_ct_MSForm As Long = 3
'This is to stop screen flashing while creating form
Application.VBE.MainWindow.Visible = False
'Add to CallerWorkbook instead
With CallerWorkbook.VBProject.VBComponents.Add(vbext_ct_MSForm)
.Properties("Caption") = "Select"
.Properties("Width") = 300
.Properties("Height") = 270
GetTempFormName = .Name
End With
End Function
Public Sub RemoveTempForm(CallerWorkbook As Workbook, FormName As String)
With CallerWorkbook.VBProject.VBComponents
Dim comp As Object
Set comp = .Item(FormName)
.Remove .Item(FormName)
End With
End Sub
然后,在用户的工作簿中添加以下代码:
Sub GetAddinToCreateForm()
Dim FormName As String
FormName = MyAddin.GetTempFormName(ThisWorkbook)
With VBA.UserForms.Add(FormName)
.Show
End With
MyAddin.RemoveTempForm ThisWorkbook, FormName
End Sub
关于vba - 在运行时将用户窗体添加到不同的工作簿,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51545083/
程序集,masm 嘿,我写了宏来打印存储在 dane1 段中的 1 字节值。 我将值除以 16,然后将提醒推送到堆栈,直到值==0。然后我弹出提醒将它们转换为 ASCII 码,并打印它们。 有人可以看
我在 Apache Geronimo 2.1.3 上有一个 Web 服务应用程序。 我正在使用 JAX-WS,使用 SOAP 1.1 注释 POJOS。 (使用 Sun JDK 1.5) 各种客户端都
我有一个数据变量,monthArray,它被多个消费者读取,并由单个定期更新程序线程定期更新数据。全部异步。 我已经考虑了这两个选项来安全地执行更新。 ArrayList tempArray
我有一组 3D 体。每个 Body 由 8 个点定义,每个点具有三个坐标。所有的物体都是立方体的或近似立方体的。我想用系统的点栅格“填充”立方体。坐标存储在简单的 data.frames 中。 我开发
我正在处理遗留代码,需要打补丁。 问题:一个古老的应用程序发送错误的 HTTP POST 请求。其中一个参数未经过 URL 编码。我知道这个参数总是排在最后而且我知道它的名字。我现在正尝试在运行在 t
我想在触摸屏幕时移动 sprite body ,但它不能发生...... -(void)ccTouchEnded:(UITouch *)touch withEvent:(UIEvent *)event
这个问题在这里已经有了答案: Can a union be initialized in the declaration? (3 个答案) 关闭 7 年前。 如果它是一个struct那么它就可以完成
我正在尝试获取生成 sigsys 信号的系统调用的地址!但我从 gcc 收到以下错误: gcc emulator.c -fms-extensions error: ‘siginfo_t’ has no
当我使用 Postman 进行 API 调用时,我收到一个 JSON 对象..这是我所期望的。 但是,当我像这样与 Guzzle 进行相同的调用时: $client = new \GuzzleHttp
在编码时,出现了差异。通常在编写简单的方法或构造函数时,我经常使用表达式体技术。但是,当我产生以下内容时: public class Sample : ISample { private r
我正在使用 LibGDX 创建一个新项目。 我想做的是,我将 tmx 文件中的主体加载到工作正常的关卡中。尸体也有一个 Sprite 。 问题是,我想让用户触摸场景中的某些 body 。当他们触摸 b
我的意图:在不使用 union 的情况下循环遍历一个结构的 30 个成员,所有成员都是字符数组类型,每个成员都存储对 itoa 的调用结果。在下面的代码中,我将结构体成员命名为a-z、A-D。在调用函
我必须将我的代码段之一从 C 转换为 java。代码如下。 union commandString{ char commndStr[20]; struct{ char
#include union NumericType { float value; int intvalue; }Values; int main() { Values.va
我在此代码中收到错误: fun num(num:Int):Int { if (num > 0){ print(num % 10) return num / 10
我是一名优秀的程序员,十分优秀!