这一章,我们来动手实践VC调用JS函数

我们动手写一个HTML,其中包含这样一段JS代码:

//[html]
<script type="text/javascript">
function Add(value1, value2)
{
return value1 + value2;
}
</script>

然后我们用WebBrowser加载这个HTML后,在VC中这样来调用这个函数名为Add的JS函数:

//[Cpp]
//别忘了#include <MsHTML.h> void CTestCallJsDlg::OnBnClickedOk()
{
//m_WebBrowser是一个WebBrowser的Activex控件对象。
CComQIPtr<IHTMLDocument2> spDoc = m_WebBrowser.get_Document();
CComDispatchDriver spScript;
spDoc->get_Script(&spScript); CComVariant var1 = , var2 = , varRet;
spScript.Invoke2(L"Add", &var1, &var2, &varRet);
}

调试(Debug):

spScript.Invoke2的作用是调用JS函数中名为Add的函数,传入两个参数,用varRet接收返回值。

可以看到,Invoke2调用成功后,varRet得到了返回值30。

但这样的话一次只能接受一个返回值。

如果要一次接受多个返回值的话,怎么办呢?

我们可以让JS返回一个JS中的Array数组或Object对象。

当JS函数return一个Array或一个Object对象时,VC这边的varRet将接受到一个代表该对象的IDispatch接口。我们仍然用CComDispatchDriver来管理这个IDispatch。用上一篇文章介绍的CComDispatchDriver的四个方法:

GetProperty

GetPropertyByName

PutProperty

PutPropertyByName

来从这个Array或Object对象中取出我们要的数据。

实践是检验真理的唯一标准,让我们再来写一个JS函数:

//[html] view plaincopy在CODE上查看代码片派生到我的代码片
<script type="text/javascript">
function Add(value1, value2) {
var array = new Array();
array[0] = value1;
array[1] = value2;
array[2] = value1 + value2;
return array;
}
</script>

然后在VC中这样写:

//[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
void CTestCallJsOnBnClickedOk()
{
CComQIPtr<IHTMLDocument2> spDoc = m_WebBrowser.get_Document();
CComDispatchDriver spScript;
  spDoc->get_Script(&spScript); CComVariant var1 = , var2 = , varRet;
spScript.Invoke2(L"Add", &var1, &var2, &varRet); CComDispatchDriver spArray = varRet.pdispVal;
//获取数组中元素个数,这个length在JS中是Array对象的属性,相信大家很熟悉
CComVariant varArrayLen;
spArray.GetPropertyByName(L"length", &varArrayLen);
//获取数组中第0,1,2个元素的值:
CComVariant varValue[];
spArray.GetPropertyByName(L"", &varValue[]);
spArray.GetPropertyByName(L"", &varValue[]);
spArray.GetPropertyByName(L"", &varValue[]);
}

调试(Debug):

可以看到,10,20,30,这三个JS函数返回的值已经躺在我们的varValue[3]里了。

当然,如果不知道JS返回的Array对象里面有几个元素,我们可以在VC这边获取它的length属性,然后在一个循环中取出数组中的每个值。

如果我们的JS函数返回一个包含有多个属性值的Object对象,VC这边该如何接收呢?

让我们再来写一个JS函数:

//[html] view plaincopy在CODE上查看代码片派生到我的代码片
<script type="text/javascript">
function Add(value1, value2) {
var data = new Object();
data.result = value1 + value2;
data.str = "Hello,我是小明!";
return data;
}
</script>

然后在VC中我们这样接收:

//[cpp]
void CTestCallJsOnBnClickedOk()
{
CComQIPtr<IHTMLDocument2> spDoc = m_WebBrowser.get_Document();
CComDispatchDriver spScript;
spDoc->get_Script(&spScript); CComVariant var1 = 10, var2 = 20, varRet;
spScript.Invoke2(L"Add", &var1, &var2, &varRet); CComDispatchDriver spData = varRet.pdispVal;
CComVariant varValue1, varValue2;
spData.GetPropertyByName(L"result", &varValue1);
spData.GetPropertyByName(L"str", &varValue2);
}

调试(Debug):

我们从JS返回的Object对象里取出了它的两个属性,result和str,分别是一个整形数据和一个字符串。

这里JS代码是我们自己写的,在VC这边当然事先知道这个JS函数返回的对象有result和str这两个属性。

如果JS代码不是我们写的,或者它的属性是事先不能确定的,该怎么办呢?答案是使用IDispatchEx接口来枚举这个对象的相关信息(方法名、属性名)。

这个现在暂时不讲,在后续的文章中会讲。

当然,JS不只可以返回Object对象,返回什么对象都可以,当返回一个对象而非基本数据类型(整形、浮点、字符串)时,VC这边收到的返回值是一个IDispatch,然后我们需要调用GetPropertyByName方法从这个IDispatch代表的对象中取出它的属性来。

这样一来,VC调用JS函数,传递参数给JS和JS返回返回值给VC,大致就都会了。

对于CComVariant包装的VARIANT这种智能型变量,不了解的可以到网上看下相关资料。《深入解析ATL》之类的书上均有介绍。

值得注意的是ATL提供的这些CCom开头的智能包装类,并不依赖于ATL的动态库。因为我在VC项目中并没有选择链接ATL,程序调试运行时进程加载的模块中也有没有ATL100.dll之类的模块载入。大家可以放心使用而不用担心依赖上ATL。

VC调用JS函数没问题了。那么JS函数如何调用VC呢?我们将在下一篇文章中慢慢道来。

最新文章

  1. 登录失败。该登录名来自不受信任的域,不能与 Windows 身份验证一起使用
  2. JS中javascript:void(0)真正含义
  3. openpgp和gnupg
  4. SQL中VARCHAR与NVARCHAR存储区别
  5. Centos 6.4 安装elasticsearch+kibana
  6. Java-反射再学习
  7. c4Droid
  8. 实体框架(Entity Framework)
  9. web前端开发初学者必看的学习路线(附思维导图)
  10. SpringBoot的几个使用技巧
  11. CorelDraw X8 破解激活问题
  12. Python序列的一点用法
  13. 08 训练Tensorflow下围棋
  14. CentOS 6、7 安装 Golang
  15. django数据库多对多修改对应关系
  16. swift 总结
  17. Python字符串颜色输出
  18. 【Unity】4.5 树木创建器
  19. Python基础教程学习(三)
  20. navicat 快速复制表所有字段

热门文章

  1. PLSQL_查询已执行SQL的绑定参数(案例)
  2. 在tortoiseSVN上将trunk的代码merge到branch上去
  3. 更改EGit的user settings中默认的location
  4. ylbtech-dbs:ylbtech-1,FAM(家庭资产管理系统)
  5. 使用LKDBHelper 插入相同id时候应该是更新数据而不是插入新的数据
  6. Java实现单向链表的增删改查
  7. js中格式化时间字符串
  8. 立体匹配:关于Middlebury提供的源码的简化使用
  9. C Primer Plus(第五版)11
  10. (medium)LeetCode 240.Search a 2D Matrix II