IE11對 Activex控制元件的相容性問題
阿新 • • 發佈:2019-01-06
;博主論述:轉載這篇文章,是為了解決公司專案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" onclick="doTest();"></input> </BODY> </HTML> 說明: codeBase="test.CAB#version=9,0,0,1" codeBase表示檔案相對或者絕對路徑;version表示版本號,如果這個號和INF檔案的版本號一樣,那麼第二次訪問頁面就不會下載,否則每次都下載。CLSID 是 ActiveX 專案生成的序號,具體可以在專案的*.rgs 檔案中找到。 好了。所有的步驟都完成了,這時你執行test.htm,提示ActiveX控制元件,你選擇允許,然後就可以呼叫加法運算了。 這只是一個簡單的例子,在其中的DLL中,你可以實現自己的應用了。