前些日子做一個Web項目,必須自己編寫一個ActiveX控件。如今的ActiveX控件大多是使用VB/C++來開發(fā)的,而我對他們并不熟悉,因此考慮使用熟悉的C#編寫ActiveX控件。
首先,建立一個WinForm控件項目HelloWorld,并拖入一個Label控件,文字設(shè)為HelloWorld,如圖:
UserControl1.cs內(nèi)容如下:
using System;
using System.Collections;
using System.ComponentModel;
using System.Drawing;
using System.Data;
using System.Windows.Forms;
namespace HelloWorld
{
/**//// <summary>
/// UserControl1 的摘要說明。
/// </summary>
public class Demo : System.Windows.Forms.UserControl
{
private System.Windows.Forms.Label label1;
/**//// <summary>
/// 必需的設(shè)計器變量。
/// </summary>
private System.ComponentModel.Container components = null;
public Demo()
{
// 該調(diào)用是 Windows.Forms 窗體設(shè)計器所必需的。
InitializeComponent();
// TODO: 在 InitComponent 調(diào)用后添加任何初始化
}
/**//// <summary>
/// 清理所有正在使用的資源。
/// </summary>
protected override void Dispose( bool disposing )
{
if( disposing )
{
if( components != null )
components.Dispose();
}
base.Dispose( disposing );
}
組件設(shè)計器生成的代碼#region 組件設(shè)計器生成的代碼
/**//// <summary>
/// 設(shè)計器支持所需的方法 - 不要使用代碼編輯器
/// 修改此方法的內(nèi)容。
/// </summary>
private void InitializeComponent()
{
this.label1 = new System.Windows.Forms.Label();
this.SuspendLayout();
//
// label1
//
this.label1.Location = new System.Drawing.Point(32, 32);
this.label1.Name = "label1";
this.label1.Size = new System.Drawing.Size(120, 32);
this.label1.TabIndex = 0;
this.label1.Text = "HelloWorld";
this.label1.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
//
// Demo
//
this.Controls.Add(this.label1);
this.Name = "Demo";
this.Size = new System.Drawing.Size(184, 96);
this.ResumeLayout(false);
}
#endregion
}
}
此時編譯項目,可以生成HelloWorld.dll。將此dll拷貝到IIS的虛擬根目錄下,然后建立一個helloworld.htm的文件,html代碼如下:
<body bgcolor=‘#223344‘>
<object id="helloworld" classid=’http://localhost/HelloWorld.dll#HelloWorld.Demo’ Width="184" Height="96" VIEWASTEXT> </object>
</body> 如圖,控件已經(jīng)成功在頁面上顯示了。OK,我們已經(jīng)完成了第一步。
但是問題到這里還沒有解決。不相信?你可以試試在另外一臺機器上測試,注意需要修改對應(yīng)的html代碼和URL地址。你可以看到這個在原來顯示控件的地方是一個紅色的叉,或者還會彈出一個對話框,表示這個控件沒有任何權(quán)限。出現(xiàn)這個結(jié)果是微軟的默認設(shè)置造成的,作者必須在控件所在的控件的 AssemblyInfo.cs/vb 中執(zhí)行一個安全聲明,聲明這個控件必須使用賦予的權(quán)限,才可以顯示出界面。我們在AssemblyInfo.cs中引用System.Security命名空間,并添加一句:
[assembly : AllowPartiallyTrustedCallers()] 現(xiàn)在重新編譯,并且替換以前的dll,界面又可以顯示出來了。
需要提醒的是,到現(xiàn)在為止,我們編寫的還不是真正的ActiveX控件。這個控件到現(xiàn)在為止,還只是能夠?qū)崿F(xiàn)自身的顯示,并且不能實現(xiàn)更多的功能,比如實現(xiàn)與腳本的交互或者操作客戶端的注冊表或者磁盤。這是由于.Net Framework的安全模型所限制的。如果我們希望這個控件突破.Net Framework安全模型的限制,實現(xiàn)與腳本的交互或者操作客戶端的注冊表或者磁盤的話,必須要讓它成為真正的ActiveX控件。下面,我們把剛才的控件變成真正的ActiveX控件。
首先使用 工具—〉創(chuàng)建GUID 生成一個GUID,并修改UserControl1.cs文件。首先增加引用System.Runtime.InteropServices命名空間,并在Demo前面加入一條語句:
注意Guid中的字符串,就是你生成的Guid字符串。它是你所生成的ActiveX控件的唯一標識符。然后修改項目屬性,如圖:
注意面板中的最后一項,我們唯一需要修改的是將其值改為True。
重新編譯。我們使用 工具—〉OLE/COM對象查看器 查看,如圖:
可以看到,我們寫的HelloWorld.Demo已經(jīng)被正確識別為COM組件?,F(xiàn)在,我們已經(jīng)可以像使用其它ActiveX控件一樣在網(wǎng)頁中顯示了。在HelloWorld.Demo點擊鼠標右鍵,如圖:
選擇Copy HTML <object> Tag to Clipboard,可以將代碼拷入剪貼板。
現(xiàn)在,我們改寫helloworld.htm,html代碼如下:
<body bgcolor=‘#223344‘>
<object id="helloworld"
classid="clsid:9551B223-6188-4387-B293-C7D9D8173E3A" Width="184" Height="96">
</object>
</body>
使用IE查看,我們的控件又可以在網(wǎng)頁中顯示了。不過,這個時候它已經(jīng)不再是以前的.net WinForm控件了,而是貨真價實的ActiveX控件了。
我們在Demo中加入ShowMessage方法:
public void ShowMessage(string msg)
{
if(msg != null)
{
MessageBox.Show(msg);
}
}
我們重新編譯。在重新訪問頁面之前,我們先來修改html代碼:
<body bgcolor=‘#223344‘>
<object id="helloworld"
classid="clsid:9551B223-6188-4387-B293-C7D9D8173E3A" Width="184" Height="96"
>
</object>
<br>
<input type=‘button‘ onclick=‘helloworld.ShowMessage(“Hello World!”)‘ value=‘Click‘>
</body>
但是結(jié)果卻很遺憾,我們發(fā)現(xiàn)IE跳出了對話框,如圖所示:
單擊確定之后,我們發(fā)現(xiàn)JS報錯。根據(jù)提示,我們判斷可以通過修改IE的設(shè)置使控件運行。打開IE的 工具——〉Internet選項——〉安全——〉本地Intranet——〉自定義級別——〉對沒有標記為安全的ActiveX控件進行初始化和運行,將其值設(shè)為啟用。我們刷新頁面,現(xiàn)在終于可以正確運行了。
當然,我們不能指望我們的客戶和我們一樣修改這個值。畢竟,一是操作麻煩,二是給電腦帶來了很大的安全風險。在互聯(lián)網(wǎng)上搜索之后,發(fā)現(xiàn)必須要實現(xiàn)IObjectSafety接口,把ActiveX控件標記為安全的ActiveX控件。在搜索MSDN之后,我找到了IObjectSafety接口的定義。這就好辦了。首先我們自己用C#實現(xiàn)這個接口:
[Guid("CB5BDC81-93C1-11CF-8F20-00805F2CD064"),InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IObjectSafety
{
// methods
void GetInterfacceSafyOptions(
System.Int32 riid,
out System.Int32 pdwSupportedOptions,
out System.Int32 pdwEnabledOptions);
void SetInterfaceSafetyOptions(
System.Int32 riid,
System.Int32 dwOptionsSetMask,
System.Int32 dwEnabledOptions);
}
[Guid("CB5BDC81-93C1-11CF-8F20-00805F2CD064"),InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IObjectSafety
{
// methods
void GetInterfacceSafyOptions(
System.Int32 riid,
out System.Int32 pdwSupportedOptions,
out System.Int32 pdwEnabledOptions);
void SetInterfaceSafetyOptions(
System.Int32 riid,
System.Int32 dwOptionsSetMask,
System.Int32 dwEnabledOptions);
}
注意,這個GUID是不能改的。然后,我們在Demo類里面實現(xiàn)這個接口。增加一下代碼:
IObjectSafety 成員#region IObjectSafety 成員
public void GetInterfacceSafyOptions(Int32 riid, out Int32 pdwSupportedOptions, out Int32 pdwEnabledOptions)
{
// TODO: 添加 WebCamControl.GetInterfacceSafyOptions 實現(xiàn)
pdwSupportedOptions = 1;
pdwEnabledOptions = 2;
}
public void SetInterfaceSafetyOptions(Int32 riid, Int32 dwOptionsSetMask, Int32 dwEnabledOptions)
{
// TODO: 添加 WebCamControl.SetInterfaceSafetyOptions 實現(xiàn)
}
#endregion
IObjectSafety 成員#region IObjectSafety 成員
public void GetInterfacceSafyOptions(Int32 riid, out Int32 pdwSupportedOptions, out Int32 pdwEnabledOptions)
{
// TODO: 添加 WebCamControl.GetInterfacceSafyOptions 實現(xiàn)
pdwSupportedOptions = 1;
pdwEnabledOptions = 2;
}
public void SetInterfaceSafetyOptions(Int32 riid, Int32 dwOptionsSetMask, Int32 dwEnabledOptions)
{
// TODO: 添加 WebCamControl.SetInterfaceSafetyOptions 實現(xiàn)
}
#endregion
重新編譯,然后將IE里面的設(shè)置改回來?,F(xiàn)在,我們發(fā)現(xiàn),和JS的交互已經(jīng)沒有問題了。
這樣,一個最基本的ActiveX控件已經(jīng)寫好了。你可以在這個控件的基礎(chǔ)上增加任何你需要的功能。到這里,編寫控件的任務(wù)已經(jīng)完成了,我們的下一個目標就是發(fā)布它。
在前面我們已經(jīng)完成了ActiveX控件的開發(fā),接下來的就是發(fā)布它了。
首先,我們建立一個windows安裝項目,并將ActiveX控件的主輸出添加到項目輸出中。然后,改動ActiveX控件的主輸出文件,將其Register屬性改為vsdrpCOM.如圖:
下一步,我們改動項目屬性,將引導程序更改為 Web引導程序。很遺憾的是,在 Web引導程序設(shè)置 中的安裝文件夾URL中必須使用絕對路徑,不能使用相對路徑。這意味著生成安裝程序的時候就必須確定路徑,不是很方便。在示例中,我使用了localhost,在發(fā)布中可以改為實際的域名。
現(xiàn)在我們生成安裝程序,并把相應(yīng)得程序拷貝到正確的目錄中(本例中為默認網(wǎng)站目錄下的ActiveX文件夾中)。我們可以直接執(zhí)行Setup.Exe文件,以驗證安裝文件的正確性。在我的機器上正確執(zhí)行了,成功了!
現(xiàn)在我們又要重新改動helloworld.htm文件了。修改后的結(jié)果如下:
<body bgcolor=‘#223344‘>
<object id="helloworld"
classid="clsid:9551B223-6188-4387-B293-C7D9D8173E3A" Width="184" Height="96" codebase="ActiveX/Setup.Exe"
>
</object>
<br>
<input type=‘button‘ onclick=‘helloworld.ShowMessage("Hello World!")‘ value=‘Click‘>
</body>
注意,我們在object塊中加入了codebase屬性,這就是制定的下載控件的位置,可以使用相對路徑。別忙,我們現(xiàn)在還不能正確請求這個頁面,因為我們還沒有對我們的控件進行簽名。簽名可以采用兩種方式,一種是在上面生成安裝程序的時候簽名,另一種是使用sn.exe簽名。推薦大家使用后者,因為可以提供更多選項。本人很懶,就不多寫了,大家可以參考csdn上的文章《發(fā)布ActiveX》。先給給大家提個醒,在申請證書的時候選擇 高級證書申請。