IE11對 Activex控件的兼容性問題

;博主論述:轉載這篇文章,是爲了解決公司項目IE11 無法使用插件ActiveX功能,解決方案:使用HTML5技術重新制作網站;跟客戶協商使用其他中和方案。
;互聯網中很多病毒都是依賴於ActiveX控件進行傳播的,微軟爲了用戶的安全性,在IE8以後的版本中逐漸取消了對ActiveX控件的支持。

ActiveX插件製作流程:
DLL封裝成ActiveX的DLL,供javascript
因項目需要,開始學習並研究VC、DLL及ActiveX控件,網上資料找了很多,但沒一個可用的或者說沒一個例子可理解並運行的。沒辦法,自己研究吧。功夫不負有心人,終有小成了,呵呵,現在把自己學習總結了一下,獻給需要的人。
一、 概述 
因項目需要,開始學習並研究VC、DLL及ActiveX控件,網上資料找了很多,但沒一個可用的或者說沒一個例子可理解並運行的。沒辦法,自己研究吧。功夫不負有心人,終有小成了,呵呵,現在把自己學習總結了一下,獻給需要的人。 
DLL(動態鏈接庫): 分WIN32 DLL和MFC DLL 
ActiveX:分ATL控件和MFC控件兩類(也是一個DLL) 
WEB:JAVASCRIPT 調用-> ActiveX調用-> DLL 完成加法運算並返回值,在頁面上顯示。 
二、開發(VS2008) 
1、DLL 庫編寫: 
文件-》新建-》WIN32控制檯->填寫項目名稱-》選擇DLL-》空項目-》完成。 
(1)在解決方案面板中,加入一個頭文件testdll.h,內容: 

複製代碼代碼如下:

#ifndef _DLLTUT_DLL_H_ 
#define _DLLTUT_DLL_H_ 
#if defined DLL_EXPORT 
#define DECLDIR __declspec(dllexport) 
#else 
#define DECLDIR __declspec(dllimport) 
#endif 
//extern "C"告訴編譯器該部分可以在C/C++中使用。 
extern "C" 
{ 
DECLDIR int Add( int a, int b ); 
DECLDIR void Function( void ); 
} 
#endif 

(2)在解決方案面板中,加入一個實現文件testdll.cpp,內容: 

複製代碼代碼如下:

#include <iostream> 
#define DLL_EXPORT 
#include "testdll.h" 
extern "C" 
{ 
// 這裏主要用到 ADD 方法。 
DECLDIR int Add( int a, int b ) 
{ 
return( a + b ); 
} 
DECLDIR void Function( void ) 
{ 
std::cout << "DLL Called!" << std::endl; 
} 
} 

(3)可選。新建一個WIN32控制檯類,測試這個DLL。 
文件-》新建-》WIN32控制檯->填寫項目名稱-》選擇控制檯程序-》空項目-》完成。 
在解決方案面板中,加入一個實現文件loaddll.cpp 內容: 

複製代碼代碼如下:

#include <iostream> 
#include <windows.h> 
using namespace std; 
typedef int (*AddFunc)(int,int); //定義指針函數、接口。 
typedef void (*FunctionFunc)(); 
int main() 
{ 
AddFunc _AddFunc; 
FunctionFunc _FunctionFunc; 
cout <<"---獲取DLL---."<< endl; 
// L 表示使用UNICODE 字符集,要和項目的字符集保持一致。 
HINSTANCE hInstLibrary = LoadLibrary(L"E:\\Project\\VS\\LoadDll\\Release\\TestDll.dll"); 
if (hInstLibrary == NULL) 
{ 
cout <<"Dll 加載【失敗】."<< endl; 
FreeLibrary(hInstLibrary); 
}else{ 
cout <<"Dll 加載【成功】."<< endl; 
} 
_AddFunc = (AddFunc)GetProcAddress(hInstLibrary, "Add"); 
_FunctionFunc = (FunctionFunc)GetProcAddress(hInstLibrary, "Function"); 
if ((_AddFunc == NULL) || (_FunctionFunc == NULL)) 
{ 
FreeLibrary(hInstLibrary);//釋放 
}else{ 
cout <<"---獲取DLL函數【OK】---."<< endl; 
} 
cout << _AddFunc(1, 1) << endl; // 開始調用 
_FunctionFunc(); // 
cin.get(); // 獲得焦點,這樣就不會程序就不會一閃而過了。 
FreeLibrary(hInstLibrary);//調用完後,要釋放內存。 
return(1); 
} 

2、ActiveX 控件實現: 
這裏我們選擇ATL控件實現,而非MFC ActiveX。 
文件-》新建-》ATL項目->填寫項目名稱(“FROMYANTAI”)-》選擇動態鏈接庫(DLL)-》完成。 
完成後,會在右邊“解決方案資源管理器”生成很多頭H文件和CPP實現文件,這些都是默認的不要修改。 
(1)、添加一個ALT簡單對象:鼠標郵件點擊項目名稱(剛纔起的名字)選擇-》添加類-》選擇ATL簡單對象。 
下一步起一個名字:“ytiicrj”—》下一步:其他不變,在支持中,選擇“連接點”和“IE對象支持”—》完成。 
下一步給“ytiicrj”添加一個方法,以便WEB頁面調用。在“類視圖”選擇“iytiicrj”(有個灰色的鑰匙圖標)鼠標右鍵添加-》添加方法。方法起名爲“GetContent”-》參數屬性選擇IN,參數類型選擇LONG 參數名 A –》添加;繼續;參數屬性選擇IN,參數類型選擇LONG 參數名 B –》添加;繼續;參數屬性選擇OUT和RETVAL ,參數類型選擇LONG* 參數名 out –》添加---》 點擊完成。 
這樣就在ytiicrj.H頭文件中添加了一個(在最後一行): 
STDMETHOD(GetContent)(LONG a, LONG b, LONG* out); 
並在ytiicrj.CPP文件中添加了一個實現類: 

複製代碼代碼如下:

STDMETHODIMP CCaluNumCtrl::GetContent(LONG a, LONG b, LONG* out) 
{ 
// TODO: 在此添加實現代碼 
return S_OK; 
} 

(2)、在ytiicrj.H 文件中,調用DLL類庫。代碼如下: 
// CaluNumCtrl.h : ytiicrj 的聲明 黑體(粗體)部分是具體的實現,其他未動。 

複製代碼代碼如下:

#pragma once 
#include "resource.h" // 主符號 
#include <windows.h> //添加 
#include "AtlActiveX_i.h" 
#include "_ICaluNumCtrlEvents_CP.h" 
#if defined(_WIN32_WCE) && !defined(_CE_DCOM) && !defined(_CE_ALLOW_SINGLE_THREADED_OBJECTS_IN_MTA) 
#error "Windows CE 平臺(如不提供完全DCOM 支持的Windows Mobile 平臺)上無法正確支持單線程COM 對象。定義_CE_ALLOW_SINGLE_THREADED_OBJECTS_IN_MTA 可強制ATL 支持創建單線程COM 對象實現並允許使用其單線程COM 對象實現。rgs 文件中的線程模型已被設置爲“Free”,原因是該模型是非DCOM Windows CE 平臺支持的唯一線程模型。" 
#endif 
// ytiicrj 
class ATL_NO_VTABLE Cytiicrj : 
//增加一下一行:安全提示解除,--當運行瀏覽器調用時,不會提示安全問題。 
public IObjectSafetyImpl<Cytiicrj, INTERFACESAFE_FOR_UNTRUSTED_CALLER| INTERFACESAFE_FOR_UNTRUSTED_DATA>, 
public CComObjectRootEx<CComSingleThreadModel>, 
public CComCoClass<Cytiicrj, &CLSID_CaluNumCtrl>, 
public IConnectionPointContainerImpl<Cytiicrj>, 
public CProxy_ICaluNumCtrlEvents<Cytiicrj>, 
public IObjectWithSiteImpl<Cytiicrj>, 
public IDispatchImpl<ICaluNumCtrl, &IID_ICaluNumCtrl, &LIBID_AtlActiveXLib, 1, 0> 
{ 
public: 
//以下三行實現定義。 
typedef int (*AddFunc)(int,int); //類型定義,對應DLL ADD方法。Func自定義,隨便寫。 
HINSTANCE hInstLibrary; 
AddFunc _AddFunc; //類映射 
Cytiicrj() 
{ 
//開始調用DLL,進行計算。 
hInstLibrary = LoadLibrary(L"TestDll.dll");//把寫好的DLL文件放在此項目生成的目錄下 
if (hInstLibrary == NULL) 
{ 
FreeLibrary(hInstLibrary);//資源釋放 
}else{ 
} 
//調用方法,返回方法句柄。 
_AddFunc = (AddFunc)GetProcAddress(hInstLibrary, "Add"); 
} 
DECLARE_REGISTRY_RESOURCEID(IDR_CALUNUMCTRL) 
BEGIN_COM_MAP(Cytiicrj) 
COM_INTERFACE_ENTRY(ICaluNumCtrl) 
COM_INTERFACE_ENTRY(IDispatch) 
COM_INTERFACE_ENTRY(IConnectionPointContainer) 
COM_INTERFACE_ENTRY(IObjectWithSite) 
//增加一下一行:安全提示解除,--當運行瀏覽器調用時,不會提示安全問題。 
COM_INTERFACE_ENTRY(IObjectSafety) 
END_COM_MAP() 
BEGIN_CONNECTION_POINT_MAP(Cytiicrj) 
CONNECTION_POINT_ENTRY(__uuidof(_ICaluNumCtrlEvents)) 
END_CONNECTION_POINT_MAP() 
DECLARE_PROTECT_FINAL_CONSTRUCT() 
HRESULT FinalConstruct() 
{ 
return S_OK; 
} 
void FinalRelease() 
{ 
FreeLibrary(hInstLibrary); 
} 
public: 
STDMETHOD(GetContent)(LONG a, LONG b, LONG* out); 
}; 
OBJECT_ENTRY_AUTO(__uuidof(CaluNumCtrl), Cytiicrj) 

(3)、回到在ytiicrj.PP 文件中,添加實現代碼如下: 

複製代碼代碼如下:

STDMETHODIMP CCaluNumCtrl::GetContent(LONG a, LONG b, LONG* out) 
{ 
// TODO: 在此添加實現代碼 
int sum = this->_AddFunc(static_cast<int>(a),static_cast<int>(b)); 
*out = static_cast<LONG>(sum); 
this->_AtlFinalRelease(); 
return S_OK; 
} 

(4)、生成DLL: 
這步很簡單,選擇 Release模式,點擊項目進行生成(會提示選擇REG32註冊,那就選擇被)。這樣就在Release目錄下生成了很多文件,我們要的就是一個DLL文件。 
3、DLL和 ATL ActiveX 控件DLL 打包爲CAB文件: 
例如:生成test.CAB後,WEB頁面就會提示下載安裝。 
(1)首先定義setup.inf文件:它描述了下載的內容和目標目錄還有版本號及相應的DLL文件。這個要手動編寫的,我的內容如下(對應名稱自行修改吧): 

複製代碼代碼如下:

[version] 
; version signature (same for both NT and Win95) do not remove 
signature="$CHICAGO$" 
AdvancedINF=2.0 
[Add.Code] 
AtlActiveX.dll=AtlActiveX.dll 
TestDll.dll=TestDll.dll 
setup.inf=setup.inf 
[install.files] 
AtlActiveX.dll=AtlActiveX.dll 
TestDll.dll=TestDll.dll 
setup.inf=setup.inf 
[AtlActiveX.dll] 
clsid={4AE870B5-C7FB-4171-A47E-7F57AFD86F67}  //clsid 接口ID (接口ID),在工程中可以找到,不可以隨便寫
file-win32-x86=thiscab 
FileVersion=1,0,0,1 
DestDir=11 
RegisterServer=yes 
[TestDll.dll] 
file-win32-x86=thiscab 
DestDir=11 
FileVersion=1,0,0,1 
RegisterServer=yes 
[setup.inf] 
file=thiscab 
[RegisterFiles] 
%11%\AtlActiveX.dll 
; end of INF file 

(2)整合資源: 
將所用到的DLL全部放到一個目錄下包括setup.inf文件,然後在開始運行:IExpress 命令去生成CAB包。 
運行後,選擇第一個,下一步,選擇第三個,下一步,添加文件(選擇你的DLL和INF文件),下一步,選擇一個輸出目錄並創建一個CAB文件名,再選擇第二個選項,下一步,選擇第二個選項,然後OK。這樣就生成了一個CAB文件。 
(3)WEB頁面調用 ActiveX 控件 進行加法運算 : 
寫一個test.htm網頁和CAB文件放在一個目錄,test.htm內容如下: 

複製代碼代碼如下:

<HTML> 
<HEAD> 
<TITLE>New Page</TITLE> 
<OBJECT id=CaluNumCtrl align="CENTER" WIDTH=0 HEIGHT=0 codeBase="test.CAB#version=9,0,0,1" classid="CLSID:B6D4B406-9CC4-4C80-B7A2-248BBB07F682"></OBJECT> 
<script language="javascript"> 
function doTest() 
{ 
var sum = CaluNumCtrl.GetContent(1,1); 
alert(sum); 
} 
</script> 
</HEAD> 
<BODY> 
<input type="button" value="renjie" id="btnOK" οnclick="doTest();"></input> 
</BODY> 
</HTML> 

說明: codeBase="test.CAB#version=9,0,0,1" codeBase表示文件相對或者絕對路徑;version表示版本號,如果這個號和INF文件的版本號一樣,那麼第二次訪問頁面就不會下載,否則每次都下載。CLSID 是 ActiveX 項目生成的序號,具體可以在項目的*.rgs 文件中找到。 
好了。所有的步驟都完成了,這時你運行test.htm,提示ActiveX控件,你選擇允許,然後就可以調用加法運算了。 
這只是一個簡單的例子,在其中的DLL中,你可以實現自己的應用了。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章