C# ACTIVEX控件开发 打包CAB安装部署 含源码及全过程开发文档

C# ACTIVEX控件开发 打包CAB安装部署 含源码及全过程开发文档
C# ACTIVEX控件开发 打包CAB安装部署 含源码及全过程开发文档

1.问题场景

在C/S架构的系统中,客户端要实现某些业务功能,可以通过安装相关的应用程序集来方便的实现。同样的需求,在B/S架构的系统里实现起来却比较困难。因为所有的程序都放在服务器端,客户端只是采用浏览器,通过HTTP协议来访问服务器端。比较成熟的解决办法是开发ActiveX控件安装到客户端,这样客户端的浏览器就可以访问本地的ActiveX控件来执行相关的本地操作。通常开发ActiveX使用C,C++或VB开发,本文将要谈论的,就是使用C#开发一个ActiveX控件。

3.注意的地方,C#开发的ActiveX控件只可在装有Framework的系统上才能用。

2.开发工具

Visual Studio2010

.NET版本:2.0

开发语言:C#

3.ActiveX控件项目创建

(1),新建一个【Windows窗体控件库项目】,项目名为ActiveXDemo

选择.NET Framwork2.0版本,因为.Net2.0安装文件只有20多M,客户端容易接受,而且如今多数家用电脑都装了.Net2.0

(2),需要更改两个地方的设置,选中刚创建的ActiveXDemo项目:

1).右键选择“属性>>应用程序>>程序集信息”设置,勾选“使程序集COM可见”:

2).右键选择“属性>>应用程序>>生成”设置,勾选“为COM互操作注册”(注意,此处如果是在debug 状态下修改的,那么在调到release状态下还需要再设置一次):

4.ActiveX控件功能添加

(1),添加一个用户控件

(2),为了使该用户控件作为一个ActiveX控件进行使用,还需要做以下修改:

首先,为控件类MiugoX.cs添加GUID,这个编号将用于B/S系统的客户端调用时使用(可以使用工具-创建GUID菜单创建一个GUID):

[Guid("DF0769A5-99D3-4BE0-90B8-91C65A92118C"),ProgId("ActiveXDemo.MiugoX"),ComVisible(true)] public partial class MiugoX:UserControl,IObjectSafety

{

……

//DF0769A5-99D3-4BE0-90B8-91C65A92118C就¨a是o?刚?刚?创???建?§的ì?Guid

(3),为了让ActiveX控件获得客户端的信任,控件类还需要实现一个名为“IObjectSafety”的接口。先创建该接口(注意,不能修改该接口的GUID值):添加一个接口IObjectSafety,实现该接口的目的就是提高程序的安全性,以便客户端IE在不更改设置的情况下可以预行该ActiveX控件。接口成员如下:using System;

using System.Runtime.InteropServices;

namespace ActiveXDemo

{

[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);

}

}

(4),用户控件MiugoX.cs需要继承并实现IObjectSafety接口,需要引用using

System.Security.Cryptography和using System.Runtime.InteropServices;两个命名空间

Miiugo.cs实现的接口成员如下:

using System;

using https://www.360docs.net/doc/2f14059858.html,ponentModel;

using System.Drawing;

using System.Windows.Forms;

using System.IO;

using System.Text;

using System.Security.Cryptography;

using System.Runtime.InteropServices;

namespace ActiveXDemo

{

[Guid("DF0769A5-99D3-4BE0-90B8-91C65A92118C"),ProgId("ActiveXDemo.MiugoX"),

ComVisible(true)]

public partial class MiugoX:UserControl,IObjectSafety

{

#region IObjectSafety成员

private const string_IID_IDispatch="{00020400-0000-0000-C000-000000000046}";

private const string_IID_IDispatchEx="{a6ef9860-c720-11d0-9337-00a0c90dcaa9}";

private const string_IID_IPersistStorage="{0000010A-0000-0000-C000-000000000046}";

private const string_IID_IPersistStream="{00000109-0000-0000-C000-000000000046}";

private const string_IID_IPersistPropertyBag=

"{37D84F60-42CB-11CE-8135-00AA004BB851}";

private const int INTERFACESAFE_FOR_UNTRUSTED_CALLER=0x00000001;

private const int INTERFACESAFE_FOR_UNTRUSTED_DATA=0x00000002;

private const int S_OK=0;

private const int E_FAIL=unchecked((int)0x80004005);

private const int E_NOINTERFACE=unchecked((int)0x80004002);

private bool_fSafeForScripting=true;

private bool_fSafeForInitializing=true;

public int GetInterfaceSafetyOptions(ref Guid riid,ref int pdwSupportedOptions,ref int pdwEnabledOptions)

{

int Rslt=E_FAIL;

string strGUID=riid.ToString("B");

pdwSupportedOptions=INTERFACESAFE_FOR_UNTRUSTED_CALLER| INTERFACESAFE_FOR_UNTRUSTED_DATA;

switch(strGUID)

{

case_IID_IDispatch:

case_IID_IDispatchEx:

Rslt=S_OK;

pdwEnabledOptions=0;

if(_fSafeForScripting==true)

pdwEnabledOptions=INTERFACESAFE_FOR_UNTRUSTED_CALLER;

break;

case_IID_IPersistStorage:

case_IID_IPersistStream:

case_IID_IPersistPropertyBag:

Rslt=S_OK;

pdwEnabledOptions=0;

if(_fSafeForInitializing==true)

pdwEnabledOptions=INTERFACESAFE_FOR_UNTRUSTED_DATA;

break;

default:

Rslt=E_NOINTERFACE;

break;

}

return Rslt;

}

public int SetInterfaceSafetyOptions(ref Guid riid,int dwOptionSetMask,int dwEnabledOptions)

{

int Rslt=E_FAIL;

string strGUID=riid.ToString("B");

switch(strGUID)

{

case_IID_IDispatch:

case_IID_IDispatchEx:

if(((dwEnabledOptions&dwOptionSetMask)==

INTERFACESAFE_FOR_UNTRUSTED_CALLER)&&

(_fSafeForScripting==true))

Rslt=S_OK;

break;

case_IID_IPersistStorage:

case_IID_IPersistStream:

case_IID_IPersistPropertyBag:

if(((dwEnabledOptions&dwOptionSetMask)==

INTERFACESAFE_FOR_UNTRUSTED_DATA)&&

(_fSafeForInitializing==true))

Rslt=S_OK;

break;

default:

Rslt=E_NOINTERFACE;

break;

}

return Rslt;

}

#endregion

}

}

(5),按照开发Windows用户控件一样的思路完成该控件的开发,本例中主要实现了三个业务功能,第一是创建文件,第二是获取某目录下的文件,第三是显示“你好,ActiveX”的提示消息。用户控件设计界面:

程序功能代码:

using System;

using https://www.360docs.net/doc/2f14059858.html,ponentModel;

using System.Drawing;

using System.Windows.Forms;

using System.IO;

using System.Text;

using System.Security.Cryptography;

using System.Runtime.InteropServices;

namespace ActiveXDemo

{

[Guid("DF0769A5-99D3-4BE0-90B8-91C65A92118C"),ProgId("ActiveXDemo.MiugoX"),ComVisible(true)] public partial class MiugoX:UserControl,IObjectSafety

{

#region IObjectSafety成员

//…IObjectSafety成员代码省略,请参考第(4)段

#endregion

public MiugoX()

{

InitializeComponent();

labelResult.Text="";

}

public void SayHelloWord()

{

MessageBox.Show("你好ActiveX!");

}

public void ShowMessage(string msg)

{

MessageBox.Show(msg);

}

public string GetHelloWord()

{

return"Hello ActiveX";

}

public void CreateText()

{

labelResult.Text="";

if(txtFileName.Text=="")

{

labelResult.Text="请输入文件名?";

return;

}

string dir="C:\\ActiveDemo";

string filePath=dir+"\\"+txtFileName.Text+".txt";

if(!Directory.Exists(dir))

{

Directory.CreateDirectory(dir);

}

if(!File.Exists(filePath))

{

File.CreateText(filePath);

labelResult.Text+=filePath+"创建成功\r\n";

}

else

{

labelResult.Text+=filePath+"已存在";

}

}

public void GetFiles()

{

string dir="C:\\ActiveDemo";

if(!Directory.Exists(dir))

{

Directory.CreateDirectory(dir);

}

string[]fileNameList=Directory.GetFiles(dir);

if(fileNameList.Length==0)

{

labelResult.Text="没有文件";

return;

}

StringBuilder sb=new StringBuilder();

foreach(string fileName in fileNameList)

{

sb.Append(fileName+"\r\n");

}

labelResult.Text="";

labelResult.Text=sb.ToString();

}

private void btnCreateFile_Click(object sender,EventArgs e)

{

CreateText();

}

private void btnGetFiles_Click(object sender,EventArgs e)

{

GetFiles();

}

}

}

(6),到这一步,一个简单的ActiveX控件已开发完成。

5.ActiveX控件调试

建立一个Web应用程序项目,在测试页面的html或aspx页面代码中添加对ActiveX控件的引用,并且可以通过Javascript调用控件的公共成员(注意这里clsid后面的值即为前面为用户控件类设置的GUID):

codebase="Install/ActiveXDemo.cab#version=1,2,0,0">


6.ActiveX控件部署和升级

ActiveX控件可以使用Visual Studio2010的安装项目进行部署。这与普通的Windows Form应用程序的部署几乎一样,只有一个地方需要注意,将前面创建的用户控件项目作为主输出项目,并设置其Register属性为vsdrpCOM,其布署和升级的详细文档请看下一篇介绍。

7.总结

综上所述,在Visual Studio2010环境中使用C#开发ActiveX控件,技术实现上没有什么难度,唯一的问题就是客户端需要安装.NET Framework。鉴于ActiveX控件一般都是实现一些简单单一的功能,.NET Framework2.0已经完全可以应付,所以建议在.NET Framework2.0下开发。因为相对于.NET Framework 3.5或4.0两百多兆的安装包,.NET Framework 2.0安装包只有20多兆,用户相对容易接受一些。

相关主题
相关文档
最新文档