在上一篇随笔: ActiveX(二)Js 监听 ActiveX中的事件  中,已经可以实现 Js 监听 ActiveX中的事件,至此、Js 和 ActiveX 已经可以实现双向通讯了。但是、这样的实现方式,都是站在Js的角度去实现的,那么 ActiveX 能否主动调用 Js 呢?答案无疑是肯定的,在该篇随笔中、我们将逐渐揭开这一层神秘的面纱。

  我第一次接触用C#代码调用Js是在四年前,那时候正在实习做Windows应用、需要借用 WebBrowser 控件操作js、完成一些特殊需求。当时的代码大致如下:

        // 执行JS
private void ExecJs()
{
try
{
if (this.webBrowser.Document != null)
{
mshtml.IHTMLDocument2 currentDoc = (mshtml.IHTMLDocument2)this.webBrowser.Document.DomDocument;
if (currentDoc != null)
{
mshtml.IHTMLWindow2 win = (mshtml.IHTMLWindow2)currentDoc.parentWindow;
if (win != null)
{
//调用 函数F、传递一个宽度做参数
win.execScript(string.Format("F('{0}')", this.webBrowser.Width), "javascript");
}
}
}
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
}

  由上述代码、可以看出,其核心方法为 mshtml.IHTMLWindow2.execScript,注意:我们再仔细看一下方法名,有没有觉得很熟悉?如果你还没想起来,那我们继续看一下 mshtml.IHTMLWindow2 这个类型中还有什么成员吧。

    [Guid("332C4427-26CB-11D0-B483-00C04FD90119")]
[TypeLibType()]
public interface IHTMLWindow2 : IHTMLFramesCollection2
{
[DispId()]
dynamic _newEnum { get; }
[DispId()]
HTMLNavigator clientInformation { get; }
[DispId()]
bool closed { get; }
[DispId()]
string defaultStatus { get; set; }
[DispId()]
IHTMLDocument2 document { get; }
[DispId()]
IHTMLEventObj @event { get; }
[DispId()]
dynamic external { get; }
[DispId()]
FramesCollection frames { get; }
[DispId()]
HTMLHistory history { get; }
[DispId()]
HTMLImageElementFactory Image { get; }
[DispId()]
int length { get; }
[DispId()]
HTMLLocation location { get; }
[DispId()]
string name { get; set; }
[DispId()]
HTMLNavigator navigator { get; }
[DispId()]
dynamic offscreenBuffering { get; set; }
[DispId(-)]
dynamic onbeforeunload { get; set; }
[DispId(-)]
dynamic onblur { get; set; }
[DispId(-)]
dynamic onerror { get; set; }
[DispId(-)]
dynamic onfocus { get; set; }
[DispId(-)]
dynamic onhelp { get; set; }
[DispId(-)]
dynamic onload { get; set; }
[DispId(-)]
dynamic onresize { get; set; }
[DispId(-)]
dynamic onscroll { get; set; }
[DispId(-)]
dynamic onunload { get; set; }
[DispId()]
dynamic opener { get; set; }
[DispId()]
HTMLOptionElementFactory Option { get; }
[DispId()]
IHTMLWindow2 parent { get; }
[DispId()]
IHTMLScreen screen { get; }
[DispId()]
IHTMLWindow2 self { get; }
[DispId()]
string status { get; set; }
[DispId()]
IHTMLWindow2 top { get; }
[DispId()]
IHTMLWindow2 window { get; } [DispId()]
void alert(string message = "");
[DispId()]
void blur();
[DispId()]
void clearInterval(int timerID);
[DispId()]
void clearTimeout(int timerID);
[DispId()]
void close();
[DispId()]
bool confirm(string message = "");
[DispId()]
dynamic execScript(string code, string language = "JScript");
[DispId()]
void focus();
[DispId()]
dynamic item(ref object pvarIndex);
[DispId()]
void moveBy(int x, int y);
[DispId()]
void moveTo(int x, int y);
[DispId()]
void navigate(string url);
[DispId()]
IHTMLWindow2 open(string url = "", string name = "", string features = "", bool replace = false);
[DispId()]
dynamic prompt(string message = "", string defstr = "undefined");
[DispId()]
void resizeBy(int x, int y);
[DispId()]
void resizeTo(int x, int y);
[DispId()]
void scroll(int x, int y);
[DispId()]
void scrollBy(int x, int y);
[DispId()]
void scrollTo(int x, int y);
[DispId()]
int setInterval(string expression, int msec, ref object language = Type.Missing);
[DispId()]
int setTimeout(string expression, int msec, ref object language = Type.Missing);
[DispId()]
void showHelp(string helpURL, object helpArg = Type.Missing, string features = "");
[DispId()]
dynamic showModalDialog(string dialog, ref object varArgIn = Type.Missing, ref object varOptions = Type.Missing);
[DispId()]
string toString();
}

IHTMLWindow2

  想起来了没有? 如果还没有、那只能说你一行Js代码都没写过。 没错,js 中的 window 对象 实现了 mshtml.IHTMLWindow2 这个接口。既然如此,我们通过一个初始化函数将 window 对象传入,就可以得到  mshtml.IHTMLWindow2 接口了。

哈哈,补充一下, mshtml 命名空间 源自:Microsoft.mshtml.dll.

当知道了如上的信息后,一切就变的‘简单’了:

        /// <summary>
/// 调用Js函数
/// </summary>
/// <param name="functionName">js函数名</param>
/// <param name="args">参数</param>
/// <returns></returns>
private dynamic RunJs(string functionName, string args = "")
{
dynamic dy = null; // 方法返回值永远为 null
if (this.window2 != null)
{
//调用函数
dy = this.window2.execScript(string.Format("{0}('{1}')", functionName, args), "javascript");
} return dy;
}

通过如上的代码,确实可以调用Js、但是、还有一个Bug... 返回值一直为Null.

那怎么办呢? 反射——没错,就是反射调用Js,也许你现在还想不明白, 先看一下代码吧,看完后你也许会恍然大悟:

        /// <summary>
/// 调用Js函数
/// </summary>
/// <param name="functionName">js函数名</param>
/// <param name="args">参数</param>
/// <returns></returns>
private dynamic RunJs(string functionName, string args = "")
{
dynamic dy = null;// 第二种方式 反射调用 Js
// 被调用的js函数必须有一个参数
if (this.window2 != null)
{
dy = this.window2.GetType().InvokeMember(functionName,
BindingFlags.Instance | BindingFlags.InvokeMethod | BindingFlags.Public,
null, this.window2, new object[] { args });
} return dy;
}

有没有想到呢 —— 全局的Js函数其实是属于window对象的,当然可以用反射去调用啦。

完整的测试项目demo: TestActiveX.zip

下一篇,我们将探究强大的 Microsoft.mshtml.dll,有点期待。

(未完待续...)

ActiveX(三)ActiveX 调用 Js的更多相关文章

  1. Unreal Engine 4 Radiant UI 插件入门(三)——从蓝图中调用JS

    不知道大家有没有混淆.这篇教程说的是从蓝图中调用JS的功能(以改变H5内的内容). 在安装了UE4和RadiantUI的前提下学习这篇教程.本篇教程接着上一篇教程,建议请先看上一篇. 第一步:在网页中 ...

  2. VC与JavaScript交互(二) --- 调用JS函数

    这一章,我们来动手实践VC调用JS函数. 我们动手写一个HTML,其中包含这样一段JS代码: //[html] <script type="text/javascript"& ...

  3. Web网页中内嵌Activex的Activex插件开发 .

    转载自: http://blog.csdn.net/tttyd/article/details/5258096 源代码下载 http://files.cnblogs.com/tttyd/Activex ...

  4. Delphi 中调用JS文件中的方法

    unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...

  5. java调用js脚本

    有些情况下,需要java去调用js,groovy等脚本语言,传入参数获取脚本运行的结果. js脚本例子: function add(a,b){ return a + b + number; } jav ...

  6. JS调用OC方法并传值,OC调用JS方法并传值////////////////////////zz

     iOS开发-基于原生JS与OC方法互相调用并传值(附HTML代码)     最近项目里面有有个商品活动界面,要与web端传值,将用户在网页点击的商品id 传给客户端,也就是js交互,其实再说明白一点 ...

  7. Magento中调用JS文件的几种方法

    一.全局调用方法: 通过该方法每个页面都会引用这个JS文件,除非是类似jQuery这样的系统文件,不然不推荐这种方法. 文件路径:/app/design/frontend/default/Your_T ...

  8. ios开发--网页中调用JS与JS注入

    先将网页弄到iOS项目中: 网页内容如下, 仅供测试: <html> <head> <meta xmlns="http://www.w3.org/1999/xh ...

  9. 服务端调用js:javax.script

    谈起js在服务端的应用,大部分人的第一反应都是node.js.node.js作为一套服务器端的 JavaScript 运行环境,有自己的独到之处,但不是所有的地方都需要使用它. 例如在已有的服务端代码 ...

随机推荐

  1. 【直播】APP全量混淆和瘦身技术揭秘

    [直播]APP全量混淆和瘦身技术揭秘 近些年来移动APP数量呈现爆炸式的增长,黑产也从原来的PC端转移到了移动端,通过逆向手段造成数据泄漏.源码被盗.APP被山寨.破解后注入病毒或广告现象让用户苦不堪 ...

  2. Windows+GCC下内存对齐的常见问题

    结构/类对齐的声明方式 gcc和windows对于modifier/attribute的支持其实是差不多的.比如在gcc的例子中,内存对齐要写成: class X { //... } __attrib ...

  3. Html5 实现灯笼绘制

    最近在学习Html5,就用JavaScript在Canvas试着绘制了一个灯笼,并作了简要的说明. 具体绘制思路在页面上有说明,不再赘述,代码如下: <script type="tex ...

  4. JQuery学习思维导图版

    常用UI资源 参考资料:Jquery教程 dataTables:教程  中文教程 Wizard:教程 Jquery UI demos:教程 selectmenu:教程 jquery-slider:教程 ...

  5. Nginx服务器之基础学习

    一.Nginx介绍 nginx:Nginx是一种软件服务器(轻量级),故它最主要的功能就是可以与服务器硬件结合,我们的应用程序可以放在nginx服务器上进行发布,已达到让网民浏览的效果.除此自外,Ng ...

  6. Ubuntu14.04、win7双系统如何设置win7为默认启动项

    Ubuntu14.04.win7双系统如何设置win7为默认启动项 Ubuntu14.04.win7双系统设置win7为默认启动项方法: 在启动项选择菜单处记住windows 7对应的序号. 从上至下 ...

  7. for循环或Repeat里面对某个字段进行复杂处理的解决方案

    在后台用一个方法处理

  8. Cordova+Asp.net Mvc+GIS跨平台移动应用开发实战1-系统初步搭建(附演示,apk,全部源码)

    1.前言 身处在移动互联网的今天,移动应用开发炙手可热,身为程序猿的我们怎么能错过开发一款我们自己的APP.本人算是一个基于.net的GIS开发入门者(马上就大四啦), 暑假在学校参加GIS比赛有大把 ...

  9. 【WP开发】JSON数据的读与写

    在不使用其他库的情况下,WP-RT应用中也可以处理JSON数据.主要的几个类都放到Windows.Data.Json命名空间中: IJsonValue接口作为用于封装JSON数据的规范,其中只读属性V ...

  10. gridview里找到控件

    ; i < gvIncomeYG.Rows.Count; i++) { Label lblYG_DYYGSR_BHS = ((Label)gvIncomeYG.Rows[i].Cells[].F ...