波斯码BOSSMA Information Technology

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

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

一个项目要用到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. 尘风 20楼:

    @bossma
    我检查了一下客户机,上面已经安装了.net 3.5 sp1 就不知道是什么原因,无法运行我的控件!

  2. 尘风 19楼:

    @bossma
    谢谢回复,太感谢了,我再去看看

  3. 尘风 18楼:

    我按照文中介绍的方法,在Microsoft Visual Studio 2008中做了一个控件,但在使用过程中发现,在装有Microsoft Visual Studio 2008的两台电脑中使用没问题,但在其他没安装Microsoft Visual Studio 2008而只安装有.net 3.5的客户机不无法运行,显示红色的叉叉,希望大家指点迷经!

  4. Terry 17楼:

    我找到版本更新的方法了

    我們需要更新cab的版本和registry的版本

    首先說說cab的版本
    cab的版本(即html裏codebase的版本)是會和客戶這兩條registry進行配對
    HKCR/CLSID/{GUID}/InstalledVersion

    HKCU/Software/Classes/{GUID}/InstalledVersion/(Default)
    也就是說,我們需要在安裝msi的時候,在客戶裏寫進這條registry
    方法如下︰
    在ActiveX Setup安?項目點擊右鍵 –> view(檢視?) –> Registry(登錄表?)裏添加
    (小弟只有英文版本,不太清楚中文版的名字,請見諒)
    建成以後,在安裝msi的時候,regisry便會被寫進客戶的機器
    (請注意,[ComRegisterFunction()]這個方法是不行的,因為安裝msi的時候不會執行)

    例如︰
    請以ActiveX控件的GUID代替以下的GUID,並請保留Registry裏的{}
    Registry: HKCR/CLSID/{GUID}/InstalledVersion/(Default) = “1,2,0,0”
    Html: classid=”clsid:GUID” codebase=”YourProduct.cab#version=1,2,0,0″

    這樣客戶在瀏覽你的網頁時,便會把html裏codebase的版本和Registry的版本進行配對,如html的版本較registry的版本新,便會下載並執行cab檔案

    第二點是msi的版本
    ActiveX Setup安?項目 –> properties(屬性?) –> version(版本?)
    這個版本要比客戶的版本較新,否則在執行msi時便會安裝失敗
    另外,也要如Net兄所言,把移除上一?版本”置?true

    以下是這兩項設置的目的︰
    1. html會和客戶的registry的版本進行配對,發現html的版本較客戶的版本新時,便會下載cab
    2. 下載cab後,便會根據inf執行msi or bat檔案,執行msi時,它會自行和上一次安裝的msi版本進行配對,今次的版本較新時,才會繼續安裝

    因此兩者也必須更新
    順帶一提,如果不寫registry,html是配對的對象會變成.net framework的版本,就是2,0,50727那個,暫時還不知原因

    以下是我的參考
    cab和registry的配對
    http://social.msdn.microsoft.com/Forums/en-US/ieextensiondevelopment/thread/48335861-2434-4058-9809-186a0bd9e320/

    在msi裏添加registry
    http://stackoverflow.com/questions/1506858/how-to-get-com-server-for-excel-written-in-vb-net-installed-and-registered-in-aut

  5. Terry 16楼:

    Terry :對不起樓上的bat語法應該是set CURDIR=%cd%@Rem msiexec.exe /i “%CURDIR%\ActiveXAlert.msi” /qn

    再次對不起…
    應該是
    set CURDIR=%cd%
    msiexec.exe /i “%CURDIR%\ActiveXAlert.msi” /qn
    @Rem代表comment

  6. Net 15楼:

    我认为第一个方法是很好的,因为要考虑到客户端机器如果没有framework框架的话,第一种方法可以设置自动下载安装framework,第二种方法则不会。至于更新的问题,你需要将你的安装项目属性中的版本号改变并且将“移除上一个版本”置为true,否则是不会自动更新的。

  7. bossma 14楼:

    @Terry
    我会尽快验证一下,希望已经解决此问题的朋友在这里留言。