波斯码BOSSMA Information Technology

C#开发ActiveX控件及制作CAB包

发布时间:2009年9月25日 / 分类:ASP.NET / 90,000 次浏览 / 评论

一个项目要用到ActiveX控件,参照网上众多的例子,ActiveX控件倒是很容易就搞出来了;
然后做了一个【安装项目】,倒是可以使用,只不过需要在客户端手动点击安装,下一步,下一步…觉得不是很爽;
又试了试【CAB项目】,好像没人用VS2008来做,费了点功夫,终于找到解决办法。
特将此经验与大家分享。

开发工具:Visual Studio 2008
.NET版本:2.0
开发语言:C#

一、编写ActiveX控件
1、新建一个【Windows窗体控件库项目】
Windows控件库项目

2、添加一个【用户控件】
添加用户控件

3、拖两个Label控件到界面上
拖两个Lable控件

4、为控件添加一个方法,用来设置在Label中显示当前时间

         //设置当前时间
      public void SetTime(string timeStr)
        {
            this.label2.Text = timeStr;
        }

5、新建一个接口:
据说实现该接口的目的就是提高程序的安全性,以便客户端IE在不更改设置的情况下可以预行该ActiveX控件。
关于这个有篇文章,推荐大家看看:
http://blog.csdn.net/optman/archive/2007/07/18/1698070.aspx

using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;

namespace HelloBossma
{
    [ComImport, GuidAttribute("CB5BDC81-93C1-11CF-8F20-00805F2CD064")]
    [InterfaceTypeAttribute(ComInterfaceType.InterfaceIsIUnknown)]
    public interface IObjectSafety
    {
        [PreserveSig]
        int GetInterfaceSafetyOptions(ref Guid riid, [MarshalAs(UnmanagedType.U4)] ref int pdwSupportedOptions, [MarshalAs(UnmanagedType.U4)] ref int pdwEnabledOptions);

        [PreserveSig()]
        int SetInterfaceSafetyOptions(ref Guid riid, [MarshalAs(UnmanagedType.U4)] int dwOptionSetMask, [MarshalAs(UnmanagedType.U4)] int dwEnabledOptions);
    }
}

6、让我们的控件实现这个接口
关于实现这个接口,篇幅较长,可以在这篇文章找到:
http://www.pinvoke.net/default.aspx/Interfaces/IObjectSafety.html

7、为我们的控件添加Guid、ProgId和ComVisible属性

namespace HelloBossma
{
    [Guid("636B8CE1-7512-464C-B63C-FC75BDCA21DB"), ProgId("HelloBossma.HelloBossmaActiveX"), ComVisible(true)]
    public partial class HelloBossmaActiveX : UserControl, IObjectSafety
    {
...
    }
}

8、打开项目属性,在生成里边,勾选【为Com互操作注册】
Com互操作注册

9、生成项目,进入到bin\Release目录下可以看到
HelloBossma.dll
HelloBossma.tlb
这两个文件时必须的,否则生成不成功。

二、制作ActiveX Setup安装文件
1、新建一个安装项目
安装项目

2、在项目上点右键,【添加】->【项目输出】,选择上边的项目
安装项目输出

3、打开安装项目的属性页面,设置安装URL
这里设置为:http://localhost/hellobossma
这个目录必须是实际存在的,用来存放生成的安装文件,供客户端下载安装。

4、生成项目
会生成两个文件,一个exe文件和一个msi文件

5、新建一个网站,添加如下代码到网页中。
其中classid即为控件的Guid。

<object classid="clsid:636B8CE1-7512-464C-B63C-FC75BDCA21DB" codebase="hellobossma/setup.exe" width="442" height="49" id="helloBossma"></object>

解压这个文件,里边有详细的使用说明。
6、将安装项目生成的exe和msi文件拷贝到网站hellobossma目录中。
HelloBossmaSetup.msi
setup.exe

7、启动网站,确保访问setup.exe的目录与【安装项目】中的【安装URL】一致。

8、一般情况下,网页会弹出提示,是否安装ActiveX控件等,如果提示了也不能安装,需要将网站添加到信任站点,并允许下载不安全和未签名控件。关于控件的签名和认证,本文不做说明,需要的请自行搜索。
安装ActiveX控件

使用这种安装方法,会在客户端弹出安装软件的界面,对客户来说,不太友好。

三、制作ActiveX CAB安装包
尝试了VS中的【CAB项目】模板,但是不幸的是,控件怎么也安装不上,放弃。
在网上看到很多人使用CABARC.EXE,试了一下,居然成功了,特将过程记录下来。

1、下载这个程序包,可以从微软下载,地址:

http://download.microsoft.com/download/platformsdk/cab/2.0/w98nt42kmexp/en-us/cabsdk.exe

2、【开始】-【运行】,输入“CMD”,在打开的命令提示符界面中,进入解压后文件的BIN目录,

这里需要加入两个文件:
HelloBossmaSetup.msi
install.inf

HelloBossmaSetup.msi 是上边的安装项目生成的安装程序。
install.inf是一个文本文件,指定运行控件所需要下载或者呈交的文件。

关于inf文件的书写规范,这里不做介绍,有需要的可以自己搜一下。
这个例子中install.inf文件的内容

[version]
signature="$CHICAGO$"
AdvancedINF=2.0

[Setup Hooks]
hook1=hook1

[hook1]
run=msiexec.exe /i "%EXTRACT_DIR%\HelloBossmaSetup.msi" /qn

有了这两个文件,运行下边的命令,成功的话会在当前目录生成HelloBossma.cab文件。
CABARC

3、把这个文件放到控件下载目录中,然后修改网页中的调用方式。

<object classid="clsid:636B8CE1-7512-464C-B63C-FC75BDCA21DB" codebase="hellobossma/HelloBossma.CAB" width="442" height="49" id="helloBossma">
</object><input onclick="helloBossma.SetTime((new Date()).getTime())" type="button" value="Click" />

这里还添加了一个按钮,点击时触发一个事件,调用控件的显示时间方法,点击按钮,你会发现控件上时间的变化。
23

OK,这篇文章到这里正文就结束了。
只是简单了解下使用C#开发ActiveX控件,很多东西还没弄清楚,有问题的话,欢迎提出来。

本博客所有文章如无特别注明均为原创。
复制或转载请以超链接形式注明转自波斯码,原文地址《C#开发ActiveX控件及制作CAB包

关键字:

建议订阅本站,及时阅读最新文章!
【上一篇】 【下一篇】

目前有37 条评论

  1. bossma 24楼:

    @ye
    需要先下载安装cabsdk.exe;然后在doc中进入cabsdk的bin目录;执行命令。cab包的名字我不确定有没有限制,你可以换换试试。

  2. ye 23楼:

    你用DOs 生成的CAB 的命令 为什么 在DOS里面报命名错误的呢

  3. bduren 22楼:

    正在做这一块的东西,虽然麻烦一些,但基本成功了! 另外我不解的是,为什么要制作CAB安装包呢,界面会更友好么?希望能够继续讨论!
    PS: 谢谢博主,受益良多!

  4. bossma 21楼:

    @尘风
    如果你的操作系统是Vista或者Win7就会出现这个问题,请看这篇文章:
    http://blog.bossma.cn/dotnet/visual-studio-2008-csharp-activex-summary-4/

  5. 尘风 20楼:

    @bossma
    由于我用到了你在总结(2)中介绍的第一种ActiveX与JavaScript交互的方法,所以我将代码分段测试,找到了原因,发现是引用 Microsoft Html Object Library(mshtml.tlb) 出了问题(客户电脑中未安装这个程序集)。
    解决办法:将该引用属性中的“复制本地”设置为“True”后,控件可以运行!

    但新的问题又出来了,网页状态栏提示以下错误:
    无法将类型为”System.__ComObject”的COM对象强制转换为类类型”mshtml.HTMLWindow2Class”
    进入CLR且不支持IProvideClassInfo或没有注册任何互操作程序集的COM组件都将包装在__ComObject类型中.这种类型的实例不能强制转换为任何其他类;不过,只要基础COM组件支持对接口IDD的QueryInterface调用,就能将这些实例强制转换为接口.

    本与博主素不相识,但在孤立无援的关键的时候竞能得到难得的帮助,再将向博主的辛勤与无私表示衷心的感谢——向博主学习、向博主致敬!

  6. 尘风 19楼:

    @bossma
    需要说明的是,在客户机上,用C# 写的其他应用程度能够正常运行,应该说.net 安装及版本是正确的!

    • 恩 那估计和安装VS2008以及.net版本没关联。我以前测试的时候,如果在VS2008中编译那个项目就会自动去注册,导致误以为能在浏览器中自动注册成功。你方便的话,可以发给我测试下。