pau*_*aul 5 .net f# exception wndproc winforms
我正在尝试使用.NET WebBrowser类型的Close事件,这似乎不是开箱即用的.(编辑:window.close()在浏览器中运行的脚本中发出调用时会发出此事件.)
一个解决方案,我看到的是,以扩展WebBrowser类override的WndProc方法.
我的扩展代码如下:
type internal ExtendedBrowser () = class
inherit System.Windows.Forms.WebBrowser()
let WM_PARENTNOTIFY : int = 0x0210
let WM_DESTROY : int = 0x0002
let closed : Event<unit> = new Event<unit>()
do ()
[<System.Security.Permissions.PermissionSet(System.Security.Permissions.SecurityAction.Demand, Name = "FullTrust")>]
override this.WndProc (msg : Message byref) =
match msg.Msg with
| wm when wm = WM_PARENTNOTIFY ->
if (not base.DesignMode) && (msg.WParam.ToInt32() = WM_DESTROY)
then closed.Trigger()
base.DefWndProc(ref msg)
| _ ->
base.WndProc(ref msg)
member this.Closed = closed.Publish
end
Run Code Online (Sandbox Code Playgroud)
这最终导致在访问该类型的实例时抛出异常:
Unhandled Exception: System.Reflection.TargetInvocationException: Unable to get the window handle for the 'ExtendedBrowser' control. Windowless ActiveX controls are not supported. ---> System.ComponentModel.Win32Exception: Error creating window handle.
at System.Windows.Forms.NativeWindow.CreateHandle(CreateParams cp)
at System.Windows.Forms.Control.CreateHandle()
at System.Windows.Forms.Control.get_Handle()
at System.Windows.Forms.WebBrowserBase.DoVerb(Int32 verb)
at System.Windows.Forms.WebBrowserBase.TransitionFromRunningToInPlaceActive()
--- End of inner exception stack trace ---
at System.Windows.Forms.WebBrowserBase.TransitionFromRunningToInPlaceActive()
at System.Windows.Forms.WebBrowserBase.TransitionUpTo(AXState state)
at System.Windows.Forms.WebBrowser.get_AxIWebBrowser2()
at System.Windows.Forms.WebBrowser.PerformNavigate2(Object& URL, Object& flag
s, Object& targetFrameName, Object& postData, Object& headers)
at System.Windows.Forms.WebBrowser.Navigate(String urlString)
at [PRODUCT].Application.WebBrowser..ctor() in C:\[PATH]\WebBrowser.fs:line 107
at Program.Main.main(String[] args) in C:\[PATH]\Program.fs:line 79
Press any key to continue . . .
Run Code Online (Sandbox Code Playgroud)
目前,它在调用Navigate("about:blank")(构建后第一次访问实例)时出错.我可以注释掉WndProc覆盖,并且工作正常(除了错过关闭事件).
有人说你必须把安全属性放在WndProc覆盖上,所以我这样做了,它没有解决问题.
其他人说你可以禁用DEP,但是我尝试了它并没有让我免除EXE.
扩展的一个实例是在浏览器的包装器中创建的(也称为WebBrowser),并且在my中创建了一个实例main,它在[STAThread]中运行(似乎也是必需的).
有谁知道会出现什么问题?
(我真正想要的是获取近距离事件通知的一种方式,所以如果有人知道另一条路线,我很乐意听到它.)
我发现一个帖子,其中其他人在重写 F# 中的 WndProc 方法时遇到了同样的问题,那里的解决方案对我有用;工作代码如下:
type ExtendedWebBrowser () = class
inherit System.Windows.Forms.WebBrowser()
let WM_PARENTNOTIFY : int = 0x0210
let WM_DESTROY : int = 0x0002
let closed : Event<unit> = new Event<unit>()
do ()
override this.WndProc (msg : Message byref) =
match msg.Msg with
| wm when wm = WM_PARENTNOTIFY ->
if (not base.DesignMode) && (msg.WParam.ToInt32() = WM_DESTROY)
then closed.Trigger()
base.DefWndProc(&msg)
| _ ->
base.WndProc(&msg)
member this.Closed = closed.Publish
end
Run Code Online (Sandbox Code Playgroud)
F# 的处理方式似乎ref与 C# 略有不同:阅读一下F# 中的参数和参数——请参阅“按引用传递”部分——看来ref调用会复制其参数并使用该副本作为引用单元格的值;因此,我传递的是一个参考单元格,其中包含内容的副本,msg而不是对原始内容的引用。地址运算符 ( ) 获取值的地址,因此这就是and方法&的参数所需的内容,而不是将它们包装在s 中。DefWndProcWndProcref