在Asp.Net頁面的開發過程中,我們肯定經常會用到自訂的UserControl來複用部分頁面元素,我們有兩種使用UserControl的方式
1、在設計時往頁面裡添加需要的UserControl(最常用的就是從SolutionExplorer拖ascx到設計頁面)
此種情況下,如果將UserControl放置在runat=server的html標籤中,將可能導致UserControl內的元素事件處理不能正確執行。
例如:我們有時用一個div作為邊框包含了需要的UserControl,而出於在運行時控制div的顯示效果(比如運行時隱藏/顯示這個div)的目的,可能將該div設為runat=server,此時,往往包含於div的UserControl內的元素事件觸發可能被忽略,UserControl內的事件處理函數往往是不能正常啟動並執行。
2、在運行時使用LoadControl函數動態載入UserControl,再Add到指定的位置
在動態載入的情況下,除了以上的問題同樣存在之外,還會帶來相應的初始化時的問題,這時,為了保證你的UserControl以你希望的語義正常運行,必須注意兩點:
1)如果在LoadControl該UserControl的同時需要調用UserControl的某個初始化函數的話,該初始化函數的調用必須在將該ctlAdd到指定的頁面某元素的Controls中之後進行,即必須按如下順序:
...
MyCtl ctl = (MyCtl)LoadControl("path");
this.someCtl.Controls.Add(ctl);
ctl.InitMethod();//本語句必須在上一條之後,否則ctl中的事件導致PostBack後,需要保持的textbox、listbox等的資料將丟失,導致各種錯誤
2)如果在LoadControl該UserControl的同時需要調用UserControl的某個初始化函數的話,該函數體內要注意對不需每次執行的代碼放入if (!IsPostBack){}語句塊中,否則會導致每次提交時被重複運行,這和Page_Load中的處理方法其實是一樣的,但往往容易被忽略。
對於動態載入的UserControl還要保證每次PostBack時都重新Load所有的UserControl(但PostBack時可以不初始化UserControl內的元素的初始值),否則事件處理函數會找不到對應的觸發源,儲存於ViewState的資料也將因找不到對應元素而保持不了。