这段时间要写一个ActiveX控件来控制扫描仪,并在扫描完成之后将文件路径通知页面。因为扫描的过程是异步的,所以我不能利用ActiveX控件公开的Scan方法来返回文件路径,结合Name Ctrl订阅联系人状态的思路,我想,如果可以用JavaScript来订阅ActiveX完成扫描的“事件”,我就可以在这个“事件”中把文件路径当作参数传递给页面了。
关于如何用c#开发ActiveX控件,兄的系列文章写的非常不错,我这里主要讲一下如何在ActiveX中调用页面上的JavaScript方法。1.引用Microsoft.mshtml
Microsoft.mshtml的路径是C:\Program Files\Microsoft.NET\Primary Interop Assemblies\Microsoft.mshtml.dll,添加引用后在ActiveX对应类中编写:
using mshtml;
2.用c#实现两个COM类,IOleClientSite和IOleContainer
[ComImport, Guid( " 00000118-0000-0000-C000-000000000046 " ), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)] public interface IOleClientSite { void SaveObject(); void GetMoniker( uint dwAssign, uint dwWhichMoniker, object ppmk); void GetContainer( out IOleContainer ppContainer); void ShowObject(); void OnShowWindow( bool fShow); void RequestNewObjectLayout(); }[ComImport, Guid( " 0000011B-0000-0000-C000-000000000046 " ), InterfaceTypeAttribute(ComInterfaceType.InterfaceIsIUnknown)] public interface IOleContainer { void EnumObjects([In, MarshalAs(UnmanagedType.U4)] int grfFlags, [Out, MarshalAs(UnmanagedType.LPArray)] object [] ppenum); void ParseDisplayName([In, MarshalAs(UnmanagedType.Interface)] object pbc, [In, MarshalAs(UnmanagedType.BStr)] string pszDisplayName, [Out, MarshalAs(UnmanagedType.LPArray)] int [] pchEaten, [Out, MarshalAs(UnmanagedType.LPArray)] object [] ppmkOut); void LockContainer([In, MarshalAs(UnmanagedType.I4)] int fLock); }
3.调用JavaScript方法
在Activex控件的对应类中就可以编写如下的CallJavaScript方法:
private void CallJavaScript( string Filenames){ Type typeIOleObject = this .GetType().GetInterface( " IOleObject " , true ); object oleClientSite = typeIOleObject.InvokeMember( " GetClientSite " , BindingFlags.Instance | BindingFlags.InvokeMethod | BindingFlags.Public, null , this , null ); IOleClientSite oleClientSite2 = oleClientSite as IOleClientSite; IOleContainer pObj; oleClientSite2.GetContainer( out pObj); // 参数数组 object [] args = new object [ 1 ]; args[ 0 ] = Filenames; // 获取页面的Script集合 IHTMLDocument pDoc2 = (IHTMLDocument)pObj; object script = pDoc2.Script; try { // 调用JavaScript方法OnScaned并传递参数,因为此方法可能并没有在页面中实现,所以要进行异常处理 script.GetType().InvokeMember( " OnScaned " , BindingFlags.Instance | BindingFlags.InvokeMethod | BindingFlags.Public, null , script, args); } catch { }}
4.在页面中实现相应的JavaScript方法
在包含这个ActiveX控件的页面中添加如下的JavaScript方法:
< script type = " text/javascript " > function OnScaned(files) { if (files) { // do something } } < / script>
这样,在ActiveX控件中调用CallJavaScript方法时,最终就会调用到页面中的OnScaned方法,藉此实现了ActiveX的“事件”机制。 在测试的过程中发现一些有趣的事情,不妨也和大家分享一下: - object元素的结束:object元素只能以<object></object>的方式结束,而不能简单的用<object/>来结束,这样结束的后果是object后边的元素都无法在JavaScript方法中获取,可能是浏览器还认为object元素没有结束吧。
- 参数的类型:最初我想在c#中给JavaScript方法传递数组类型的参数,但当JavaScript方法执行时,我发现JavaScript将该参数识别为“unknown”,并且无法对其做任何处理,所以最后只好作罢,用传递以“|”分隔的字符串代替。
- 提供了更简单的调用JavaScript方法,只需要在初始化时传递页面的window属性,但我还没弄清楚如何给JavaScript传递参数。