ocx开发经验

来源:互联网 发布:淘宝网找工作 编辑:程序博客网 时间:2024/05/22 08:16

http://blog.csdn.net/xiaoxiaoyu85/article/details/6821205


1. ocx下载解压的目标位置,由注册表项指定

HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Internet Settings\ActiveX Cache

默认为C:\WINDOWS\Downloaded Program Files

inf文件也解压到该目录

下载的ocx在资源管理器中不可见,使用dir命令可见。可以使用copy命令将ocx拷贝到指定文件夹,就可以看ocx信息了

 

2.VS2005开发的ocx,客户端需要环境  vcredist_x86\vcredist_x86.exe

 

3.inf书写

   (1) cab中文件(*dll; *.ini等)解压的目标目录由DestDir标示,其中

        DestDir=11  C:\windows\System 

        DestDir=10  C:\windows\ 

 

4.WIN7 + IE8中使用activeX

1.activeX要实现安全接口,否则IE8会拦截控件方法

2.要以管理员权限运行IE,然后再输入IP进行访问主页,否则会出现DEP拦截,IE会崩溃.

3.未数字签名的控件加载需要添加可信站点.

 

如何关闭IE8当前安全设置会使计算机有风险提示
IE8的安全提示真的很烦人,如使用工行网银就会提示“当前安全设置使计算机有风险。请单击这里更改安全设置……” 下面介绍关闭此提示的步骤: 1.点击 开始→运行→输入gpedit.msc 打开组策略编辑器。 2.依次点击 计算机配置→管理模板→Windows组件→Internet Explorer 在右边找到“关闭安全设置检查功能”。 3.双击“关闭安全设置检查功能”进行设置,选择“已启用”单击确定。 关闭所有的IE窗口,重新运行就不会再显示“当前安全设置会使计算机有风险提示……”的提示了。
可能有些系统里面根本就找不到那个项,没办法,只好找到组策略相应的键值更改注册表了(关闭IE):
  a.开始 -> 运行 -> regedit
  b.到HKLM\Software\Policies\Microsoft\Internet Explorer\Security键下,如果这些键不存在,自己新建,在新建过程中注意大小写。
  c.在上面的键下面新建一个双字(DWORD)型值项(Value Entry):DisableSecuritySettingsCheck,赋于键值:"1"(无引号)。同样,在新建值项的时候注意大小写。
 
 
5.IE8下需要实现安全接口
/**************** 
// 在ActiveX的类声明文件XXXCtrl.h的头部中添加如下代码 
#include <objsafe.h> 

// 在ActiveX的类声明文件C..Ctrl.h的类声明内部添加如下代码 
// 接口映射 
DECLARE_INTERFACE_MAP() 
BEGIN_INTERFACE_PART(ObjSafe, IObjectSafety) 
STDMETHOD_(HRESULT, GetInterfaceSafetyOptions) ( 
REFIID riid, 
DWORD __RPC_FAR *pdwSupportedOptions, 
DWORD __RPC_FAR *pdwEnabledOptions 
); 

STDMETHOD_(HRESULT, SetInterfaceSafetyOptions) ( 
REFIID riid, 
DWORD dwOptionSetMask, 
DWORD dwEnabledOptions 
); 
END_INTERFACE_PART(ObjSafe); 
*********************************************** *******************/ 

/********************** 
// 在ActiveX的类实现文件XXXCtrl.cpp中添加如下代码,并将CXXXCtrl更换为你的ActiveX的类名称 
// IObjectSafety的接口映射 
BEGIN_INTERFACE_MAP( CXXXCtrl, COleControl ) 
INTERFACE_PART(CXXXCtrl, IID_IObjectSafety, ObjSafe) 
END_INTERFACE_MAP() 


//////////////////////////////////// ///////////////////////////////////////// 
// IObjectSafety 成员函数 
// AddRef, Release, QueryInterface 
ULONG FAR EXPORT CXXXCtrl::XObjSafe::AddRef() 

METHOD_PROLOGUE(CXXXCtrl, ObjSafe) 
return pThis->ExternalAddRef(); 


ULONG FAR EXPORT CXXXCtrl::XObjSafe::Release() 

METHOD_PROLOGUE(CXXXCtrl, ObjSafe) 
return pThis->ExternalRelease(); 


HRESULT FAR EXPORT CXXXCtrl::XObjSafe::QueryInterface( 
REFIID iid, void FAR* FAR* ppvObj) 

METHOD_PROLOGUE(CXXXCtrl, ObjSafe) 
return (HRESULT)pThis->ExternalQueryInterface(&iid, ppvObj); 


const DWORD dwSupportedBits = 
INTERFACESAFE_FOR_UNTRUSTED_CALLER | 
INTERFACESAFE_FOR_UNTRUSTED_DATA; 
const DWORD dwNotSupportedBits = ~ dwSupportedBits; 

/////////////////////////////////////////////////////// ////////////////////// 
// CXXXCtrl::XObjSafe::GetInterfaceSafetyOptions 
HRESULT STDMETHODCALLTYPE 
CXXXCtrl::XObjSafe::GetInterfaceSafetyOptions( 
REFIID riid, 
DWORD __RPC_FAR *pdwSupportedOptions, 
DWORD __RPC_FAR *pdwEnabledOptions 


METHOD_PROLOGUE(CXXXCtrl, ObjSafe) 

HRESULT retval = ResultFromScode(S_OK); 

// 接口是否存在 
IUnknown FAR* punkInterface; 
retval = pThis->ExternalQueryInterface(&riid, (void * *)&punkInterface); 
if (retval != E_NOINTERFACE) // 接口存在 

punkInterface->Release(); // 释放引用 


// We support both kinds of safety and have always both set, 
// regardless of interface. 
*pdwSupportedOptions = *pdwEnabledOptions = dwSupportedBits; 
return retval; // E_NOINTERFACE if QI failed 


//////////////////////////////////////////////////////////// ///////////////// 
// CXXXCtrl::XObjSafe::SetInterfaceSafetyOptions 
HRESULT STDMETHODCALLTYPE 
CXXXCtrl::XObjSafe::SetInterfaceSafetyOptions( 
REFIID riid, 
DWORD dwOptionSetMask, 
DWORD dwEnabledOptions 


METHOD_PROLOGUE(CXXXCtrl, ObjSafe) 

// 接口是否存在 
IUnknown FAR* punkInterface; 
pThis->ExternalQueryInterface(&riid, (void * *)&punkInterface); 
if (punkInterface) // 接口存在 

punkInterface->Release(); // 释放引用 

else // 接口不存在 

return ResultFromScode(E_NOINTERFACE); 

// Can't set bits we don't support. 
if (dwOptionSetMask & dwNotSupportedBits) 

return ResultFromScode(E_FAIL); 


// Can't set bits we do support to zero 
dwEnabledOptions &= dwSupportedBits; 
// (We already know there are no extra bits in mask. ) 
if ((dwOptionSetMask & dwEnabledOptions) != dwOptionSetMask) 

return ResultFromScode(E_FAIL); 


// Don't need to change anything since we're always safe. 
return ResultFromScode(S_OK); 
}
 
6.在OCX中使用GDI+
GDI+的初始化放在控件Ctrl类中,不能放在APP类中。
如果OCX中有对话框类作为Ctrl的成员变量, 那么对话框类先构造,然后再构造控件类;
如果对话框类中使用到了GDI+类的指针(比如new 了Image *, Bitmap *指针对象),那么我们最终肯定是要delete它们的,否则有内存泄露, 这个时候,因为Ctrl类先析构,在析构的时候已经将GDI+的类库信息释放了,所以我们再在对话框类中delete一个GDI+对象指针会出错。
我的做法是: 在控件类析构函数中,在释放GDI+前,调用一个函数,来清理对话框中GDI+对象指针.
 
 
7.关于添加可信站点,说真的,实在不想分享,自己探索了很久的
      ocx打成cab包后让用户自动下载,会弹出安装提示; 有时候cab很大,下载也需要时间;当各个页面都有ocx时,多个cab很多会提示多次安装; 所以我们可以将ocx做成安装包,在客户端安装后, 修改IE的安全等级 ,将 “仅允许经过批准的域在未经提示的情况下使用activeX” 改为禁用, 将 "允许运行以前未使用的activeX控件而不提示"改为启用 .就需要添加可信站点等操作了.
     对应的installshiedl12脚本如下:
   function int changreg()  
    NUMBER nRootKey,nType,nSize;
    STRING szKey,szNumName, szOcxVersion;
    STRING szNumValue; 
    int res;
begin    
    //启用IE的对未可信的ActiveX初始化...选项 
    nRootKey = HKEY_CURRENT_USER;
    RegDBSetDefaultRoot (nRootKey);//置根为nRootKey       
    
     nType=REGDB_NUMBER;
     nSize=4;
     szKey="Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Zones\\3";
     
     szNumName = "1208";    
     szNumValue = "0";
     res = RegDBSetKeyValueEx(szKey,szNumName,nType,szNumValue,nSize);
     if(res < 0) then 
        MessageBox("修改IE注册表出错!", WARNING); 
     endif; 
     
     szNumName = "1209";    
     szNumValue = "0";
     res = RegDBSetKeyValueEx(szKey,szNumName,nType,szNumValue,nSize);
     if(res < 0) then 
        MessageBox("修改IE注册表出错!", WARNING); 
     endif;  
     
     szNumName = "120B";    
     szNumValue = "0";
     res = RegDBSetKeyValueEx(szKey,szNumName,nType,szNumValue,nSize);
     if(res < 0) then 
        MessageBox("修改IE注册表出错!", WARNING); 
     endif;
 
8.修改ocx的CLSID和名称
使用Create GUID工具,创建一个IMPLEMENT_OLECREATE的GUID, 创建后生成GUID替换如下位置的GUID.
IMPLEMENT_OLECREATE_EX(CMultiwndCtrl, "MULTIWND.MultiwndCtrl.1",
0x69ed7568, 0xd5a3, 0x49ea, 0x84, 0xd0, 0x8c, 0x20, 0x13, 0x67, 0x27, 0x24)
 
将16进制的GUID去掉 0x符号并进行拆分,替换原来的UUID如下
[ uuid(69ed7568-d5a3-49ea-84d0-8c2013672724)
你看到其中的规律了吧。
 
 
9.修改ocx的名称
这个最简单,在工程属性Linker - General 的output中,修改生成的ocx名称就可以了
 
10.在容器中动态改变控件的大小

在控件类的OnSize()中添加如下代码(否则鼠标点击控件就变成默认大小):

 CRect   windowRect; 
 GetWindowRect(&windowRect); 
 CWnd*   pWnd   =   GetParent(); 
 if(pWnd) 
 { 
  pWnd-> ScreenToClient(&windowRect); 
  SetRectInContainer(&windowRect); 
 }


然后在容器窗口中使用::SetWindowPos()改变控件的大小和位置即可

 

11.修改控件初始化的默认大小

     (1) 重载控件类的虚函数OnSetExtent(LPSIZEL lpSizeL),直接return false;

     (2) 在控件的Ctrl类的构造函数中使用:  this->SetInitialSize(500, 200);即可设置默认的大小

 

12.

近日把IE升级到8,结果发现在VC2005下新建类的窗口出错,不能正常使用,网上搜来解决办法,把如下文本拷到一个文本文件中,改扩展名为reg,双击导入注册表即可。

Windows Registry Editor Version 5.00

[HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Internet Settings\Zones\1000]

“1207″=dword:00000000

VS2008貌似也有同样问题,解决方法相同。

 

13.IE8无法在网页中调试ocx

关闭IE的保护模式,避免IE用多进程方式打开网页,否则调试器启动的IE进程跟网页所在的IE进程不同,就不能中断。

通过修改注册表改为多进程工作的时候,启动的IE进程和加载要调试的ocx的IE进程不是一个进程,所以不能调试

1. 点击【开始】->【运行】 命令:regedit.
2. 定位到HKEY_LOCALMACHINE -> SOFTWARE -> Microsoft -> Internet Explorer -> Main
3. 在【右边区域】【右键】新建一个名称为TabProcGrowth的DWORD值, 数值数据设置为0.


1 0