From: http://icrwen.iteye.com/blog/1118766
http://blog.csdn.net/yimiyangguang1314/article/details/6563540
将ocx和DLL文件打包成cab文件,实现IE浏览器在线安装
我们打开淘宝等网站时,IE浏览器会提示安装空间,这个控件便是用于对用户名密码进行加密的ActiveX控件。如何在我们的站点上安装如此控件,让用户可以通过简单的点击便可方便使用我们的空间呢?
下面是如何让在你的站点上安装控件的方法。
基础知识
html语言中Object标签
定义和用法
定义一个嵌入的对象。请使用此元素向您的 XHTML 页面添加多媒体。此元素允许您规定插入 HTML 文档中的对象的数据和参数,以及可用来显示和操作数据的代码。<object> 标签用于包含对象,比如图像、音频、视频、Java applets、ActiveX、PDF 以及 Flash。object 的初衷是取代 img 和 applet 元素。不过由于漏洞以及缺乏浏览器支持,这一点并未实现。浏览器的对象支持有赖于对象类型。不幸的是,主流浏览器都使用不同的代码来加载相同的对象类型。而幸运的是,object 对象提供了解决方案。如果未显示 object 元素,就会执行位于 <object> 和 </object> 之间的代码。通过这种方式,我们能够嵌套多个 object 元素(每个对应一个浏览器)。
实例
向HTML中添加对象:
- <object classid="clsid:F08DF954-8592-11D1-B16A-00C0F0283628" id="Slider1"
- width="100" height="50">
- <param name="BorderStyle" value="1" />
- <param name="MousePointer" value="0" />
- <param name="Enabled" value="1" />
- <param name="Min" value="0" />
- <param name="Max" value="10" />
- </object>
参考
http://www.w3school.com.cn/tags/tag_object.asp
Windows中注册类ID:clsid
“通用唯一标识符”(UUID),用于标识 COM 组件。每个 COM 组件在 Windows 注册表中都有自己的 CLSID,以便让其他应用程序加载。
ActiveX控件CLSID的获取
在编译完成每一个ActiveX控件之后,系统会生成ocx.html文件,其中是调用该ActiveX控件的方法。该页面中有一个标签即为Object,其中有一行则记录着该ocx文件的CLSID,如下:
- <object
- ID="TopoSceneViewer"
- WIDTH=800
- HEIGHT=600
- CLASSID="CLSID:321B84C8-3745-4AC9-BF04-55753F40CE63">
- </object>
创建cab文件
cab文件中需要包含一系列文件,其中.inf文件会告诉IE如何去安装这些文件。我们从.inf文件开始
怎么写inf
INF文件的组成有节(Sections),键(Key)和值(value)三部分。 关键节有
- [Version]版本描述信息,主要用于版本控制。
- [Strings]字符串信息,用于常量定义。
- [DestinationDirs]定义系统路径信息。
- [SourceDisksNames]指明源盘信息。
- [SourceDisksNames]指明源盘文件名。
- [DefaultInstall]开始执行安装。
其它的节可以自定义
最开始一般是[Version]区:
- [Version]
- signature="$XXXX$"
- AdvancedINF=2.0
"Signature"项定义了该INF文件需要运行在何种操作系统版本中。有$Windows NT$, $Chicago$, or $Windows 95$三个值供选择,一般选择$Chicago$即可。
接下来就是最重要的[Add.Code]区:
- [Add.Code]
- Ctrl1.dll=C1Section
- Ctrl2.dll=Ctrl2.dll
前面是要下载的文件名,后面是对应这个文件的区域名,可以是任何名字,不过一般都是和文件的名字相同,这样方便维护。
还有需要注意是在[Add.Code]区出现的文件要根据依赖性进行排序,例如前面说的ctrl1.dll要依赖于ctrl2.dll,则ctrl2.dll要出现在ctrl1.dll的前面。因为安装时是按照相反的顺序进行的,也就是说先安装ctrl2.dll,然后才是ctrl1.dll,哧哧,记清楚了,不要搞反了。
再接下来是各个文件的区域了
- [Ctrl1.dll]
- file-win32-x86=thiscab
- RegisterServer=yes
- clsid={.....}
- DestDir=
- FileVersion=1,0,0,0
[Ctrl1.dll]区域中的第一个file值告诉ie到哪里去得到这个dll,file一共包括三个部分,第一部分是file,这个永远都是这样的(至少目前来说);第二部分告诉声明支持的OS,win32表示windows,mac就是苹果MAC OX了;第三部分是CPU类型,比如说x86、 ppc (Power PC)、 mips或者alpha了。
file的值可以取三个一个URL、ignore和thiscab,如果是URL则说明到URL所在的位置去下;如果是ignore说明对于这种OS和CPU,不需要下载这个文件(ctrl1.dll);如果是thiscab很明显就在当前的cab文件中了。
接下来是RegisterServer,可以取两个值yes和no,如果为yes则说明ie要注册该dll,如果是no就不必了。+如果这里选yes,则需要指定clsid,否则clsid一行可以省略。+
再下来是DestDir,它的值是dll将要存到本地硬盘的位置,如果它的值是10,则将dll放到/Windows或者/WinNT下;如果是11,则放到/Windows/System或者/WinNT/System32下;如果是空(就是没有值)则会放到/Windows或者/WinNT下的Downloaded Program Files目录下;
最后是FileVersion,这个就比较明显了,说明了ctrl1.dll的版本号。
其他部分详见参考文献。
创建cab文件
- 如果需要创建cab文件,首先需要Cabarc或者Makecab,它们随着Cabinet SDK的安装就有了,Cabinet SDK的下载地址是http://msdn.microsoft.com/workshop/management/cab/cabdl.asp
- Cabarc可以创建、查看或者解出cab里面的文件,而Makecab则只可以用来创建cab文件。
- 制作cab文件时需要将所有的相关文件都包含进去,可以通过Depends(VC自带的)检查需要的文件。使用inf文件将这些东西都写进去。
- inf搞法:inf文件描述cab中所有的ocx及dll文件,inf通过一些命名区域来提供需要的信息。
N表示要创建一个新的文件,ctrl1.cab是创建的文件名,ctrl1.inf是cab的inf,后而是需要加到cab里的文件,可以使用通配符。- cabarc N ctrl1.cab ctrl1.inf ctrl1.dll
然后就可以将cab文件放到网页上了
参考
- 如何编写INF:http://www.cnblogs.com/sbdx/archive/2006/12/14/whatisinffile.html
- 如何写cab相关的INF:http://docs.google.com/Doc?docid=0Aehzv88zbfy_ZGdncGdqczZfMjIxZHNnZDhtYzg&hl=zh_CN
签名
申请证书
要使用到makecert工具,在VS2005目录下搜索可以搜索到。
在命令提示行中执行
- makecert -sk myNewKey -r -n "CN=发证机关的名字,OU=Certification,O=公司名字,E=name@email.com" -ss my myNew.cer
- cert2spc myNew.cer myNew.spc
makecert具体使用帮助可参考创建证书的工具(makecert.exe)
签名
要使用到signtool工具,也可以在VS2005目录下找到。
signtool工具有多种使用模式,下面使用的是带UI的向导模式。
在命令提示行中执行
- signtool signwizard
之后进入signcodewizard:
- step1.选择要签名的cab文件
- step2.自定义
- step3.从文件中选择 myNew.spc
- step4.csp中的私钥/密钥容器:myNewKey
- step5.sha1
- step6.next
- step7.描述:插件名称/web:www.yourcompany.com
- step8.http://timestamp.verisign.com/scripts/timstamp.dll
signtool 的具体使用帮助可参考签名工具 (SignTool.exe)
ActiveX控件打包cab时INF文件的编写
虽然微软对ActiveX越来越不信任,但是这个技术还是很多人喜欢用的,这个博客的编辑器貌似就用到了ActiveX。
在我们编写好ocx控件后只有在打包成CAB文件加入数字签名后才能正常发布,如果你用的是vc6,那么问题简单了,打包完成后不用考虑控件的运行库问题,现在的XP系统和将要淘汰的2kpro都已经有了这样的运行库。但是如果使用的是.net2005就会遇到有关运行库的问题,这些问题只能依靠CAB压缩包中的安装文件 .inf来解决。
首先看看基本的inf文件格式
;区域设置
signature="$CHICAGO$"
AdvancedINF=2.0
[Add.Code]
;该项会指明该安装包中有哪些项,这里说明包中有ActiveX.ocx、msvcr80.dll、mfc80u.dll、msvcp80.dll 四个项
;这里一般会使用文件名和项名相同的方法,主要是便于维护
ActiveX.ocx=ActiveX.ocx
msvcr80.dll=msvcr80.dll
mfc80u.dll=mfc80u.dll
msvcp80.dll=msvcp80.dll
[ActiveX.ocx]
;这就是Activex控件文件了
file-win32-x86=thiscab
;表示所需要的这个文件就在当前的安装包中,file-win32-x86表示该文件是一个win32系统下的文件
clsid={A3B4C60F-24A7-40E8-A06B-1A5EFE5F0A43}
;这里是这个安装包的GUID这个值是在创建项目的时候系统生成在,一般存放在以项目名命名的idl文件中,使用项目的类信息的GUID
RegisterServer=yes
;说明这个控件是需要注册的,
;一般情况下只有编写的文件是需要注册的,运行库文件和系统中可能存在的DLL文件是不用注册的,如MFC70.dll等
FileVersion=1,0,0,1
;这个文件的版本,注意要用逗号隔开
[msvcr80.dll]
;这是ocx控件的第一个依赖文件,有的地方说明依赖文件必须按照依赖顺序安装,可以用"所剩dll都依赖的文件最先安装”的规则来进行,但是我并没有在MSDN上看到什么地方有很明确的说明,并且在我自己的测试中也没有发现有关顺序的问题,并且很多地方都是将ocx控件写在最前面的,但是ocx文件是最依赖其他文件的,貌似按顺序写的做法有点“迷信”
FileVersion=8,00,50727,42
;这里是该dll文件的版本,这里注意下,这个版本可以在文件属性的版本选项卡中看到,不知道为什么这里显示的版本号有时会比最外面显示的版本号多一个0,注意,复制到inf文件后要将点分隔符改成逗号分隔符
hook=mfc80installer
;这个hook是个保留字,说明该项目被挂接到哪里,如果有则表示被挂接到目标项,hook不是一个必填项,这里表示该文件将被挂接到mfc80installer项
[mfc80u.dll]
FileVersion=8,00,50727,42
hook=mfc80installer
[msvcp80.dll]
FileVersion=8,00,50727,42
hook=mfc80installer
;上面两项都跟[msvcr80.dll]一样,这里就不解释了
[mfc80installer]
;挂接项
file-win32-x86=http://192.168.1.117/helloactivex/mfc80.cab
;这里表示这个被挂接的文件的位置在一个地址
run=%EXTRACT_DIR%vcredist_x86.exe
;这里表示从这个地址下载到cab包后执行包里面的vcredist_x86.exe文件,%EXTRACT_DIR%表示下载后的解压目录
上面有一点没有说明白,就是“为什么会出现hook这个东西?”,很多的inf文件是这样写的
msvcr80.dll=msvcr80.dll
mfc80u.dll=mfc80u.dll
msvcp80.dll=msvcp80.dll
ActiveX.ocx=ActiveX.ocx
[msvcr80.dll]
file-win32-x86=thiscab
;该文件在当前安装包中
FileVersion=8,00,50727,42
RegisterServer=no
;不需要注册
destdir=11
;该文件将被安装到系统所在目录的system32下
;........
我使用这样的方法在没有安装过任何软件的2kpro和XP下测试过,均安装失败,其原因就是运行库不匹配,如果你的控件是用VS2005的MFC开发的话,使用这样的方法无法将msvcr80.dll,mfc80u.dll,msvcp80.dll这几个控件依赖的文件安装到目标位置的,具体是什么原因我也不明白。
这里的hook就是为了解决上面说的问题:要给被按装该控件的机器安装一个运行环境。很多人可能都看过http://support.microsoft.com/kb/167158/这里说明了如何制作一个inf文件,但是很可惜,微软对mfc的下载支持到VC 6.0就结束了,并且,这里还有一个错误,在他的主要示例中有一段
file-win32-x86=VALUE=http://activex.microsoft.com/controls/vc/mfc42.cab
上面的“VALUE=”是没有必要的,如果增加了该内容反而会让安装包无法下载。
有了在依赖文件dll的项中有了挂接的hook的值后安装文件会到hook的位置去找需要下载安装的文件,这里我将/Microsoft Visual Studio 8/SDK/v2.0/BootStrapper/Packages/vcredist_x86/vcredist_x86.exe文件做成一个cab文件MFC80.cab,这个包里面只有一个vcredist_x86.exe文件,没有其他文件。
按照上面的方法打包好mfc8的运行库并使用hook的方式将Dll和运行库文件挂接后就可以了。
以上内容在没有安装任何软件的XP和2k pro上测都通过了,顺便说下,Activex控件,只要在控件测试的那个工具中跑成了就是好的,如果目标机器上出不来就是环境问题,inf是解决环境问题的唯一途径。
最后补充一点,上面说的都是在发布Release版本的情况,如果是Debug版本就会少两个库文件“MFC80D.DLL”和“MSVCR80D.DLL”
vc下使用activex控件
http://222.30.226.10/hhcmc/study/teach_vc/teach_sp_24.htm