微軟樣本:
就像您可以通過編程方式在 Web Form頁上建立任意 ASP.NET 伺服器控制項的執行個體,您也可以通過使用包含頁的 LoadControl 方法來做到這一點。但您首先必須使用@ Control 指令的className 屬性將強型別與使用者控制項相關聯。之所以需要這樣是因為LoadControl 方法返回 Control 類的類型,並且您需要將該使用者控制項轉換為合適的強型別,以便設定該控制項的各個屬性。
以下代碼使用 className 屬性將 MyUserControl.ascx 檔案中儲存的使用者控制項轉換為強型別。
<%@ Control className="MyUserControl" %>
以編程方式建立使用者控制項的執行個體
- 使用@ Reference 指令在要包含使用者控制項的 Web Form頁的頂部註冊該使用者控制項。當以編程方式建立使用者控制項時,只有您建立了對該控制項的引用後,使用者控制項的強型別才可用於該 Web Form頁。例如,以下代碼建立對 MyUserControl.ascx 檔案中所建立使用者控制項的引用。
<%@ Reference Control="MyUserControl.ascx" %>
注意 當以聲明方式在 Web Form頁中建立使用者控制項的執行個體時,請使用@ Register 指令。
- 在程式碼後置類別檔案中,或是在包含 .aspx 檔案的代碼聲明塊中,建立使用者控制項的執行個體。根據需要分配屬性值,並使用 Add 方法將該控制項添加到包含頁的 ControlCollection 對象上。這使該控制項可用於該頁的繼承的 Control.Controls 屬性。在以下樣本中,建立 MyUserControl.ascx 的執行個體並將其 BackColor 屬性設定為 beige。
[C#]Control c1 = LoadControl("MyUserControl.ascx");((MyUserControl)c1).BackColor = "beige";Page.Controls.Add(c1);[Visual Basic]Dim c1 As UserControl = LoadControl("MyUserControl.ascx")CType(c1, MyUserControl).BackColor = "beige"Page.Controls.Add(c1)
注意 當您使用 Add 方法將控制項添加到 ControlCollection 對象時,這些控制項將按處理順序放置在集合中。如果您希望將控制項添加到集合中的特定位置,請使用 AddAt 方法並指定您要儲存該控制項的索引位置。
地址:
ms-help://MS.VSCC/MS.MSDNVS.2052/cpguide/html/cpconinstantiatingusercontrolsprogrammatically.htm
鴻雪樣本:
註: 本方法是我同事李強原創
步驟一: 把ascx控制項拖入設計視窗,然後轉入html模式,保留定義
如果你需要多個ascx的話,請全部拖入。
但是把相應的內容刪除。(因為我們需要動態載入)
在需要填入控制項的地方,放個容器,比如td,並設定在伺服器端運行
如
<TD id="tdpan" runat=server></TD>
2:動態調用
UserControl myusercontrol = (UserControl) LoadControl ("../includes/pageNavigater.ascx") ;
Type myusertype = myusercontrol.GetType();
//下面是給ascx賦值
PropertyInfo myuserinfo1 = myusertype.GetProperty("RelatedDatagrid");//) .GetProperty("RelatedDatagrid");
myuserinfo1.SetValue(myusercontrol,gridhwcy ,null);
PropertyInfo mypassinfo = myusertype.GetProperty("torefresh");
mypassinfo.SetValue(myusercontrol,true,null );
PropertyInfo myuserdatasource = myusertype.GetProperty("RelatedDataSource");
myuserdatasource.SetValue(myusercontrol,dv,null);
tdpan.Controls.Clear();
tdpan.Controls.Add(myusercontrol);
如果還有什麼不明白,請和 H.xue@163.net 聯絡
地址:
http://www.dev-club.com/club/bbs/showAnnounce.asp?id=1506069
ASP.net中動態載入使用者控制項
UserControl control = (_pager.LoadControl(String.Format("~/WebUC/{0}.ascx", _ctrlId))) as UserControl;
control.ID = "uc_" + _ctrlId;
_panelControl.Controls.Add(control);
ASP.net中動態載入使用者控制項時一些問題的總結
經常見到有人說在ASP.net中不要使用動態控制項,我想主要的原因在於使用動態控制項會帶來一些問題,在做項目的過程中,我將由動態載入控制項引發的總是作了一個小小的總結。
1、在使用LoadControl載入控制項後,使用者控制項中的某些控制項不再響應事件。
這個問題主要是由於將控制項載入放在if (!Page.IsPostBack)之內引起的,放在外面即可。在思歸的blog上對此問題進行了詳細的說明。
2、使用者控制項中某些控制項的響應出現問題,如某個按鈕第一次選擇時不觸發CLICK事件,第二次可以了。
這是由於沒有給控制項設定ID引起的,控制項ID的作用在下面詳細講述。 如
Control userControl=(Control)Page.LoadControl(“Test.ascx”);
userControl.ID=“Test”;
AddControl(userControl);
3、如果使用者控制項中包括DataGrid控制項,那麼載入控制項後可能出現不響應DataGrid事件的問題。
這好像是一個bug,必須要將載入的控制項進行強制轉換,如:
Test userControl=(Test)Page.LoadControl(“Test.ascx”);
注意:上面使用的是Test類型,而不是Control!
我在以前的Blog中曾提到過這個問題,這種方式將使系統的擴充性降低。 我有一個解決方案可以和大家討論(運用策略模式):
public class BaseControl : System.Web.UI.UserControl
{
public virtual BaseControl ProcessThisControl();
}
所有的使用者控制項從BaseControl 繼承,如果有Datagrid控制項,由overide ProcessThisControl方法,如:
return this as Test;
按如下方式載入控制項:
BaseControl userControl=(BaseControl )Page.LoadControl(“Test.ascx”);
userControl.ProcessThisControl();
4、在使用者控制項中如何使用JavaScript
大家都知道,使用用戶端的指令碼將大大提高頁面的響應速度,同時可以避免頻繁地重新整理頁面。所以使用javascript來實現頁面中部分控制是一個比較好的方式,但是在使用者控制項中如果訪問某一個子控制項呢?
使用方式如下:
document.all.<%= TestControl.ClientID%><%= lstUser.ClientID%>.disabled=true; //將TestControl設定為不可用
如果在C#指令碼中應該這樣寫:
Page.RegisterStartupScript("OnInitControl","<SCRIPT LANGUAGE='JavaScript'>document.all.Test_TestControl.disabled=true;</SCRIPT>"); //Test為使用者控制項,TestControl為使用者控制項中的子控制項。
現在說一下控制項ID,在訪問aspx檔案時,IIS會將aspx的指令碼進行編譯。編譯的時候將使用者控制項中的內容寫在同一個頁面中,為了防止頁面中的控制項與使用者控制項中的控制項名稱相同,在編譯的時候對使用者控制項中的控制項名稱修改為 : 使用者控制項名:子控制項 ,控制項ID則修改為 使用者控制項ID_子控制項ID。在動態載入控制項時,如果不對控制項的ID進行賦值,則控制項ID為上一次載入的控制項ID,因此在載入使用者控制項後應該立即對其設定ID。
自訂控制項類:
using System;using System.Data;using System.Configuration;using System.Web;using System.Web.Security;using System.Web.UI;using System.Web.UI.WebControls;using System.Web.UI.WebControls.WebParts;using System.Web.UI.HtmlControls;namespace CatChen.Cocollab.Web.UI.Controls{ public class ViewPanel : System.Web.UI.Control, System.Web.UI.INamingContainer { private string _virtualPath; private Control _view; public string VirtualPath { get { return this._virtualPath; } set { string oldValue = this._virtualPath; this._virtualPath = value; if (this.Page != null && value != oldValue) { this.ChildControlsCreated = false; this.EnsureChildControls(); } } } public Control View { get { return this._view; } } protected override void OnInit(EventArgs e) { base.OnInit(e); if (!string.IsNullOrEmpty(this.VirtualPath)) { this.EnsureChildControls(); } } protected override void LoadViewState(object savedState) { Pair pair = savedState as Pair; if (pair != null) { base.LoadViewState(pair.First); this.VirtualPath = pair.Second as string; } } protected override void CreateChildControls() { this.Controls.Clear(); if (string.IsNullOrEmpty(this.VirtualPath)) { return; } if (this.Page == null) { throw new Exception("ViewPanel.Page is null."); } this._view = this.Page.LoadControl(this.VirtualPath); if (this._view == null) { throw new Exception("ViewVirtualPath cannot be loaded."); } this._view.ID = "ucView"; this.ClearChildState(); this.Controls.Add(this._view); } protected override object SaveViewState() { return new Pair(base.SaveViewState(), this.VirtualPath); } }}