ocx开发经验,ActiveX开发快速入门

各种BHO,ActiveX,chrome扩展插件技术交流  v: youyouran2017

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

   其实依赖的是4个dll,包含在安装报中,拷贝到system32也是也是一样的

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.

设置窗口初始化大小

 window.moveTo(200, 200);//移动窗口   
window.resizeTo(400, 300);//改变大小
//动态设置ocx控件大小
var obj = document.getElementById("multiwnd");
 obj.setAttribute("height", "200");

Windows Registry Editor Version 5.00

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

“1207″=dword:00000000

各种BHO,ActiveX,chrome扩展插件技术交流,youyouran2017

bat以管理员权限运行,注册ocx

@echo off
>nul 2>&1 "%SYSTEMROOT%\system32\cacls.exe" "%SYSTEMROOT%\system32\config\system"
if '%errorlevel%' NEQ '0' (
goto UACPrompt
) else ( goto gotAdmin )
:UACPrompt
echo Set UAC = CreateObject^("Shell.Application"^) > "%temp%\getadmin.vbs"
echo UAC.ShellExecute "%~s0", "", "", "runas", 1 >> "%temp%\getadmin.vbs"
"%temp%\getadmin.vbs"
exit /B
:gotAdmin
if exist "%temp%\getadmin.vbs" ( del "%temp%\getadmin.vbs" )
pushd "%CD%"
CD /D "%~dp0"

regsvr32 -s FileReader.ocx

VC2015添加ocx事件

在控件类上右键鼠标,添加事件即可

win7, win10 64位IE使用64位ocx

组合键win+R打开运行,然后输入gpedit.msc打开计算机的本地组,在管理模块中的所有设置中找到

设置选项卡进程增长,长度填0,就可以使用了

注册64位,相关的dll拷贝到system32

@echo off

copy *.ocx "C:\Windows\SysWOW64"

regsvr32 "C:\Windows\SysWOW64\YuOcx.ocx"

注意: 32位的应用程序,只能添加32位的ocx,如32位的excel, VC2015(是32位ide)等,只能添加32位的ocx,(也可以用导入ocx方式加入64位ocx,但是不方便)

编译了x64的ocx,要想在程序中添加,应该也用x64,如VC2022就是64位的ide, 可以插入64位ocx

excel插入ocx控件后,默认不触发OnCreate事件,excel开发模式中,设置ocx的熟悉为AutoLoad,下次打开就会自动触发

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值