001
|
1.在COM组件中调用JavaScript函数
|
002
|
// 连接点方式页面javascript脚本
|
003
|
<object classid=
"CLSID:B568F111-DFE4-4944-B67F-0728AB2AB30F"
id=
"testCom"
VIEWASTEXT></object>
|
004
|
<script language=
"JavaScript"
for
=
"testCom"
event=
"staTe(s)"
>
|
005
|
alert(
"State("
+ s +
")"
);
|
006
|
return
123;
|
007
|
</script>
|
008
|
<script language=
"JavaScript"
>
|
009
|
testCom.FireStateEvent(
"Hello"
);
|
010
|
</script>
|
011
|
012
|
// 事件属性方式页面javascript脚本
|
013
|
function onState(s){
|
014
|
alert(
"onState("
+ s +
")"
);
|
015
|
return
456;
|
016
|
}
|
017
|
var o =
new
ActiveXObject(
"TestATL.TestCom"
);
|
018
|
o.onstaTe=onState;
|
019
|
o.FireStateEvent(
"Hello"
);
|
020
|
021
|
// Com组件VC7.1 ATL代码
|
022
|
__interface _ITestComEvents{
|
023
|
[id(1), helpstring(
"State事件"
)]
HRESULT
State([in] BSTR str);
|
024
|
};
|
025
|
__event __interface _ITestComEvents;
|
026
|
IDispatchPtr m_onState;
// 事件属性
|
027
|
STDMETHOD(get_onState)(IDispatch** pVal) {
|
028
|
*pVal = m_onState;
|
029
|
return
S_OK;
|
030
|
};
|
031
|
STDMETHOD(put_onState)(IDispatch* newVal) {
|
032
|
m_onState = newVal;
|
033
|
return
S_OK;
|
034
|
};
|
035
|
STDMETHOD(FireStateEvent)(BSTR str) {
|
036
|
__raise State(str);
// 激发连接点事件
|
037
|
CComVariant result;
|
038
|
CComVariant avarParams[1] = {str};
|
039
|
DISPPARAMS dispParams = {avarParams, NULL, 1, 0};
|
040
|
EXCEPINFO excepInfo;
|
041
|
memset
(&excepInfo, 0,
sizeof
excepInfo);
|
042
|
UINT
nArgErr = (
UINT
)-1;
// initialize to invalid arg
|
043
|
if
(m_onState)
// 激发属性事件
|
044
|
HRESULT
hr = m_onState->Invoke(0, IID_NULL, LOCALE_USER_DEFAULT,
|
045
|
DISPATCH_METHOD, &dispParams, &result, &excepInfo, &nArgErr);
|
046
|
return
S_OK;
|
047
|
}
|
048
|
参见:
|
049
|
How To Call a Script Function from a VC WebBrowser Application
|
050
|
如何在COM object中使用 Javascript function object?
|
051
|
在COM组件中调用JavaScript函数
|
052
|
053
|
2.从页面javascript向Com组件传递结构数组
|
054
|
// 页面脚本
|
055
|
var o =
new
ActiveXObject(
"TestATL.TestCom"
);
|
056
|
o.onstaTe=onState;
|
057
|
o.Put(
"array"
, {0: 123, 1:
"abc"
});
|
058
|
o.Put(
"array"
, [456,
"def"
]);
|
059
|
o.Put(
"array"
, [{name:
"tom"
, age: 8}, {name:
"jack"
, age: 10}]);
|
060
|
var a =
new
Array(789,
"ghi"
);
// has "length" property
|
061
|
o.Put(
"array"
, a);
|
062
|
063
|
// Com组件VC7.1 ATL代码
|
064
|
STDMETHODIMP CTestCom::Put(BSTR key, VARIANT value)
|
065
|
{
|
066
|
WCHAR
output[4096] = L
""
;
|
067
|
if
(0 == wcsicmp(key, L
"array"
) && VT_DISPATCH == value.vt)
|
068
|
{
|
069
|
IDispatchPtr spDisp = value.pdispVal;
|
070
|
DISPID dispID = 0;
|
071
|
DISPPARAMS dispParams = {NULL, NULL, 0, 0};
|
072
|
CComVariant result;
|
073
|
EXCEPINFO excepInfo;
|
074
|
memset
(&excepInfo, 0,
sizeof
excepInfo);
|
075
|
UINT
nArgErr = (
UINT
)-1;
// initialize to invalid arg
|
076
|
unsigned
int
length = 0;
// 数组长度 或 属性 个数
|
077
|
078
|
LPOLESTR func = L
"length"
;
|
079
|
HRESULT
hr = spDisp->GetIDsOfNames(GUID_NULL, &func, 1, LOCALE_SYSTEM_DEFAULT, &dispID);
|
080
|
if
(S_OK == hr){
// 如果有"length"属性
|
081
|
hr
= spDisp->Invoke(dispID, IID_NULL, LOCALE_USER_DEFAULT,
DISPATCH_PROPERTYGET, &dispParams, &result, &excepInfo,
&nArgErr);
|
082
|
if
(S_OK == hr && VT_I4 == result.vt)
|
083
|
length = result.intVal;
// 直接读取数组长度
|
084
|
}
else
{
|
085
|
unsigned
int
nTypeInfo = 0;
|
086
|
hr = spDisp->GetTypeInfoCount(&nTypeInfo);
|
087
|
ATLASSERT(1 == nTypeInfo);
|
088
|
ITypeInfoPtr spTypeInfo;
|
089
|
hr = spDisp->GetTypeInfo(0, 0, &spTypeInfo);
|
090
|
TYPEATTR *pTypeAttr = NULL;
|
091
|
hr = spTypeInfo->GetTypeAttr(&pTypeAttr);
|
092
|
//ATLASSERT("{C59C6B12-F6C1-11CF-8835-00A0C911E8B2}" == pTypeAttr->guid); // JScript:
|
093
|
length = pTypeAttr->cVars;
// 从类型信息读取数组长度
|
094
|
spTypeInfo->ReleaseTypeAttr(pTypeAttr);
|
095
|
}
|
096
|
for
(unsigned
int
i=0; i<length; i++)
|
097
|
{
|
098
|
WCHAR
buf[32];
|
099
|
_itow(i, buf, 10);
|
100
|
func = buf;
|
101
|
hr = spDisp->GetIDsOfNames(GUID_NULL, &func, 1, LOCALE_SYSTEM_DEFAULT, &dispID);
|
102
|
hr
= spDisp->Invoke(dispID, IID_NULL, LOCALE_USER_DEFAULT,
DISPATCH_PROPERTYGET, &dispParams, &result, &excepInfo,
&nArgErr);
|
103
|
if
(S_OK != hr)
|
104
|
continue
;
|
105
|
if
(VT_DISPATCH == result.vt){
|
106
|
IDispatchPtr spItem = result.pdispVal;
|
107
|
func = L
"name"
;
|
108
|
hr = spItem->GetIDsOfNames(GUID_NULL, &func, 1, LOCALE_SYSTEM_DEFAULT, &dispID);
|
109
|
hr
= spItem->Invoke(dispID, IID_NULL, LOCALE_USER_DEFAULT,
DISPATCH_PROPERTYGET, &dispParams, &result, &excepInfo,
&nArgErr);
|
110
|
if
(S_OK == hr && VT_BSTR == result.vt)
|
111
|
swprintf(output + wcslen(output), L
"name=%s"
, result.bstrVal);
|
112
|
func = L
"age"
;
|
113
|
hr = spItem->GetIDsOfNames(GUID_NULL, &func, 1, LOCALE_SYSTEM_DEFAULT, &dispID);
|
114
|
hr
= spItem->Invoke(dispID, IID_NULL, LOCALE_USER_DEFAULT,
DISPATCH_PROPERTYGET, &dispParams, &result, &excepInfo,
&nArgErr);
|
115
|
if
(S_OK == hr && VT_I4 == result.vt)
|
116
|
swprintf(output + wcslen(output), L
" age=%d/n"
, result.intVal);
|
117
|
}
else
if
(VT_BSTR == result.vt)
|
118
|
swprintf(output + wcslen(output), L
"BSTR:%s/n"
, result.bstrVal);
|
119
|
else
if
(VT_I4 == result.vt)
|
120
|
swprintf(output + wcslen(output), L
"I4:%d/n"
, result.intVal);
|
121
|
else
|
122
|
swprintf(output + wcslen(output), L
"item.vt=%d/n"
, result.vt);
|
123
|
}
|
124
|
}
|
125
|
FireStateEvent(output);
|
126
|
return
S_OK;
|
127
|
}
|
128
|
129
|
3.枚举IE窗口的内容,并调用其中的脚本
|
130
|
#import <mshtml.tlb> // Internet Explorer 5
|
131
|
#import <shdocvw.dll>
|
132
|
SHDocVw::IShellWindowsPtr spSHWinds;
|
133
|
spSHWinds.CreateInstance(__uuidof(SHDocVw::ShellWindows));
|
134
|
long
nCount = spSHWinds->GetCount();
|
135
|
IDispatchPtr spDisp;
|
136
|
for
(
long
i = 0; i < nCount; i++)
|
137
|
{
|
138
|
_variant_t va(i, VT_I4);
|
139
|
spDisp = spSHWinds->Item(va);
|
140
|
SHDocVw::IWebBrowser2Ptr spBrowser(spDisp);
|
141
|
if
(spBrowser != NULL)
|
142
|
{
|
143
|
_bstr_t location = spBrowser->GetLocationName();
|
144
|
if
(_bstr_t(L
"Test DapCtrl"
) == location)
// 找指定IE窗口
|
145
|
{
|
146
|
IHTMLDocument2Ptr spDoc(spBrowser->GetDocument());
|
147
|
if
(spDoc != NULL)
|
148
|
{
|
149
|
_bstr_t
exp
= m_onState;
|
150
|
IDispatch *pdis = NULL;
|
151
|
hr = spDoc->get_Script(&pdis);
|
152
|
if
(pdis){
|
153
|
DISPID tmpDispID = 0;
|
154
|
LPOLESTR func = L
"Test"
;
// javascript 函数名
|
155
|
hr = pdis->GetIDsOfNames(GUID_NULL, &func, 1, LOCALE_SYSTEM_DEFAULT, &tmpDispID);
|
156
|
if
(S_OK == hr)
|
157
|
hr = pdis->Invoke(tmpDispID, IID_NULL, LOCALE_USER_DEFAULT,
|
158
|
DISPATCH_METHOD, &dispParams, &result, &excepInfo, &nArgErr);
|
159
|
}
|
160
|
}
|
161
|
}
|
162
|
}
|
163
|
}
|
164
|
参见:
|
165
|
HOWTO: Connect to a Running Instance of Internet Explorer
|
166
|
ActiveX组件与JavaScript交互
|
167
|
ActiveX组件控制其所在的IE窗口
|
168
|
169
|
4.在VC中执行脚本
|
170
|
#import <msscript.ocx> // msscript.ocx
|
171
|
using
namespace
MSScriptControl;
|
172
|
IScriptControlPtr pScriptControl(__uuidof(ScriptControl));
|
173
|
LPSAFEARRAY psa;
|
174
|
SAFEARRAYBOUND rgsabound[] = { 1, 0 };
// 1 elements, 0-based
|
175
|
int
i;
|
176
|
psa = SafeArrayCreate(VT_VARIANT, 1, rgsabound);
|
177
|
if
(!psa)
|
178
|
{
|
179
|
return
E_OUTOFMEMORY;
|
180
|
}
|
181
|
VARIANT vFlavors[1];
|
182
|
for
(i = 0; i < 1; i++)
|
183
|
{
|
184
|
VariantInit(&vFlavors[i]);
|
185
|
V_VT(&vFlavors[i]) = VT_BSTR;
|
186
|
}
|
187
|
V_BSTR(&vFlavors[0]) = SysAllocString(bstr);
|
188
|
long
lZero = 0;
|
189
|
hr = SafeArrayPutElement(psa, &lZero,&vFlavors[0]);
|
190
|
for
(i=0;i<1;i++)
|
191
|
{
|
192
|
SysFreeString(vFlavors[i].bstrVal);
|
193
|
}
|
194
|
pScriptControl->Language =
"JScript"
;
|
195
|
pScriptControl->AllowUI = TRUE;
|
196
|
_bstr_t
exp
= L
"1+2+3"
;
|
197
|
_variant_t outpar = pScriptControl->Eval(
exp
);
|
198
|
//_variant_t outpar = pScriptControl->ExecuteStatement(exp);
|
199
|
//_variant_t outpar = pScriptControl->Run("MyStringFunction", &psa);
|
200
|
_bstr_t bstrReturn = (_bstr_t)outpar;
|
201
|
char
*pResult = (
char
*)bstrReturn;
|
202
|
SafeArrayDestroy(psa);
|
203
|
参见:
|
204
|
http://www.cnblogs.com/zdxster/archive/2011/01/27/1945879.html |