Visual C#.NET是微軟公司出品的一種新的編程語言(以下簡稱C#),它繼承了C語言的一些特性,也加入了一些新的元素。以前用過Delphi開發(fā)程序的人可能剛開始使用C#的時候,對其有一種似曾相識的感覺(至少包括我)。是的,C#語言的創(chuàng)始人正是以前在Borland公司開發(fā)出Delphi語言的Anders Hejlsberg。在我開始使用C#開發(fā)程序時,就覺得它是一款很棒的開發(fā)Windows Form & Web程序的RAD工具。
在開發(fā)Web程序方面,C#的出現(xiàn)打破了以前的網(wǎng)頁開發(fā)模式,實現(xiàn)了與開發(fā)Windows
Form程序一樣的所見即所得的功能。C#提供了一些常用的Web Form Control供開發(fā)人員使用,并且只需將控件拖入頁面中即可,非常簡單。但有時這些控件也不能滿足開發(fā)人員的需要,需要開發(fā)人員自己編寫用戶控件(User Control)或自定義控件(Custom Control)來滿足需求。在這里,我將講解如何在C#中開發(fā)服務(wù)器控件。
二、預(yù)備知識
在C#中可以開發(fā)兩種服務(wù)器控件,一個是用戶控件(User Control)和自定義控件(Custom Control)。用戶控件的本質(zhì)與頁面文件(aspx文件)差不多,是可被其它aspx頁面重復(fù)使用的HTML代碼段,當(dāng)然它也包括后臺代碼(Code-behind),后綴名是ascx。所以在開發(fā)一些公用的靜態(tài)頁面時(例如頁頭,頁腳)經(jīng)常用到,但它的缺點是不易繼承,不易分發(fā),無法編譯成二進制代碼來進行部署。但是自定義控件的功能就強大許多,它可以被編譯成二進制代碼(DLL文件),可以被擴展、繼承、分發(fā)。就像Web Form Control一樣,其實它們每個控件就是一個DLL文件。
開發(fā)用戶控件比較簡單,就像編寫一個aspx頁面一樣,在這里就不介紹了。本文對象是自定義控件。服務(wù)器控件的基類是System.Web.UI.Control。如果要開發(fā)可視化的服務(wù)器控件,那我們需要從System.Web.UI.WebControls來繼承,否則從System.Web.UI.Control繼承。
服務(wù)器控件在設(shè)計時以runat=”server”腳本代碼嵌入到aspx文件中來表示此控件是在服務(wù)器端運行的。在服務(wù)器控件所在頁面提交回傳(PostBack)過程中是依靠ViewState(視圖狀態(tài))來維護控件狀態(tài)的。所以我們在設(shè)計服務(wù)器控件屬性時,其值應(yīng)保存在ViewState中。
三、代碼編寫
C#中有一個日歷控件Calendar,但是現(xiàn)在我需要一個可以下拉的日歷控件,并且初始時不顯示日歷,當(dāng)我點擊下拉按鈕時才彈出,并且當(dāng)選擇了日期,日歷會自動隱藏且選擇的日期值會顯示到相應(yīng)的輸入框中。顯然Calendar控件不能滿足我的需要,但是稍后我會在我的自定義控件中用到它。
首先新建項目,在項目類型中選擇Visual C#項目,在模板列表中選擇Web控件庫,輸入項目名稱AquaCalendar,然后選擇項目所在目錄,點擊【確定】按鈕。C#將會生成基本的框架代碼。將項目中的類文件和類名改名為DatePicker(即日期控件的類名)。由于DatePicker是可視化控件,所以我們必須從System.Web.UI.WebControls繼承。并且它包括一個輸入框,一個按鈕和日歷控件,需要在DatePicker類中聲明它們。像這種以多個服務(wù)器控件組合的控件成為復(fù)合控件。代碼如下,比較重要的方法和代碼在注釋中會加以說明:
using System; using System.Web.UI; using System.Web.UI.WebControls; using System.ComponentModel; using System.Drawing; namespace AquaCalendar { [DefaultProperty("Text"), //在屬性工具箱中顯示的默認(rèn)屬性 ToolboxData("<{0}:DatePicker runat=server></{0}:DatePicker>")] public class DatePicker : System.Web.UI.WebControls.WebControl , IPostBackEventHandler { //選擇日期按鈕的默認(rèn)樣式 private const string _BUTTONDEFAULTSTYLE = "BORDER-RIGHT: gray 1px solid; BORDER-TOP: gray 1px solid; BORDER-LEFT: gray 1px solid; CURSOR: hand; BORDER-BOTTOM: gray 1px solid;"; //按鈕默認(rèn)文本 private const string _BUTTONDEFAULTTEXT = "..."; private System.Web.UI.WebControls.Calendar _Calendar; public override ControlCollection Controls { get { EnsureChildControls(); //確認(rèn)子控件集都已被創(chuàng)建 return base.Controls; } } //創(chuàng)建子控件(服務(wù)器日歷控件) protected override void CreateChildControls() { Controls.Clear(); _Calendar = new Calendar(); _Calendar.ID = MyCalendarID; _Calendar.SelectedDate = DateTime.Parse(Text); _Calendar.TitleFormat = TitleFormat.MonthYear; _Calendar.NextPrevFormat = NextPrevFormat.ShortMonth; _Calendar.CellSpacing = 0; _Calendar.Font.Size = FontUnit.Parse("9pt"); _Calendar.Font.Name = "Verdana"; _Calendar.SelectedDayStyle.BackColor = ColorTranslator.FromHtml("#333399"); _Calendar.SelectedDayStyle.ForeColor = ColorTranslator.FromHtml("White"); _Calendar.DayStyle.BackColor = ColorTranslator.FromHtml("#CCCCCC"); _Calendar.TodayDayStyle.BackColor = ColorTranslator.FromHtml("#999999"); _Calendar.TodayDayStyle.ForeColor = ColorTranslator.FromHtml("Aqua"); _Calendar.DayHeaderStyle.Font.Size = FontUnit.Parse("8pt"); _Calendar.DayHeaderStyle.Font.Bold = true; _Calendar.DayHeaderStyle.Height = Unit.Parse("8pt"); _Calendar.DayHeaderStyle.ForeColor = ColorTranslator.FromHtml("#333333"); _Calendar.NextPrevStyle.Font.Size = FontUnit.Parse("8pt"); _Calendar.NextPrevStyle.Font.Bold = true; _Calendar.NextPrevStyle.ForeColor = ColorTranslator.FromHtml("White"); _Calendar.TitleStyle.Font.Size = FontUnit.Parse("12pt"); _Calendar.TitleStyle.Font.Bold = true; _Calendar.TitleStyle.Height = Unit.Parse("12pt"); _Calendar.TitleStyle.ForeColor = ColorTranslator.FromHtml("White"); _Calendar.TitleStyle.BackColor = ColorTranslator.FromHtml("#333399"); _Calendar.OtherMonthDayStyle.ForeColor = ColorTranslator.FromHtml("#999999"); _Calendar.NextPrevFormat = NextPrevFormat.CustomText; _Calendar.NextMonthText = "下月"; _Calendar.PrevMonthText = "上月"; _Calendar.Style.Add("display","none"); //默認(rèn)不顯示下拉日歷控件 _Calendar.SelectionChanged += new EventHandler(_Calendar_SelectionChanged); this.Controls.Add(_Calendar); } [ Category("Appearance"), //該屬性所屬類別,參見圖 DefaultValue(""), //屬性默認(rèn)值 Description("設(shè)置該日期控件的值。") //屬性的描述 ] public string Text { get { EnsureChildControls(); return (ViewState["Text"] == null)?System.DateTime.Today.ToString("yyyy-MM-dd"):ViewState["Text"].ToString(); } set { EnsureChildControls(); DateTime dt = System.DateTime.Today; try { dt = DateTime.Parse(value); } catch { throw new ArgumentOutOfRangeException("請輸入日期型字符串(例如:1981-04-29)!"); } ViewState["Text"] = DateFormat == CalendarEnum.LongDateTime?dt.ToString("yyyy-MM-dd"):dt.ToString("yyyy-M-d"); } } //重載服務(wù)器控件的Enabled屬性,將選擇日期按鈕變灰(禁用) public override bool Enabled { get { EnsureChildControls(); return ViewState["Enabled"] == null?true:(bool)ViewState["Enabled"]; } set { EnsureChildControls(); ViewState["Enabled"] = value; } } public string ButtonStyle { get { EnsureChildControls(); object o = ViewState["ButtonSytle"]; return (o == null)?_BUTTONDEFAULTSTYLE:o.ToString(); } set { EnsureChildControls(); ViewState["ButtonSytle"] = value; } } [ DefaultValue(CalendarEnum.LongDateTime), ] public CalendarEnum DateFormat { get { EnsureChildControls(); object format = ViewState["DateFormat"]; return format == null?CalendarEnum.LongDateTime:(CalendarEnum)format; } set { EnsureChildControls(); ViewState["DateFormat"] = value; DateTime dt = DateTime.Parse(Text); Text=DateFormat == CalendarEnum.LongDateTime?dt.ToString("yyyy-MM-dd"):dt.ToString("yyyy-M-d"); } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string MyCalendarID //復(fù)合控件ID { get { EnsureChildControls(); return this.ClientID+"_MyCalendar"; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string MyCalendarName //復(fù)合控件名稱 { get { EnsureChildControls(); return this.UniqueID+":MyCalendar"; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string DatePickerInputID //復(fù)合控件中輸入框的ID { get { EnsureChildControls(); return this.ClientID+"_DateInput"; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string DatePickerInputName //復(fù)合控件中輸入框的名稱 { get { EnsureChildControls(); return this.UniqueID+":DateInput"; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string DatePickerButtonID //復(fù)合控件中按鈕的ID { get { EnsureChildControls(); return this.ClientID+"_DateButton"; } } [ Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden) ] public string DatePickerButtonName //復(fù)合控件中按鈕的名稱 { get { EnsureChildControls(); return this.UniqueID+":DateButton"; } } public string ButtonText { get { EnsureChildControls(); return ViewState["ButtonText"] == null?_BUTTONDEFAULTTEXT:(string)ViewState["ButtonText"]; } set { EnsureChildControls(); ViewState["ButtonText"] = value; } } /// <summary> /// 將此控件呈現(xiàn)給指定的輸出參數(shù)。 /// </summary> /// <param name="output"> 要寫出到的 HTML 編寫器 </param> protected override void Render(HtmlTextWriter output) { //在頁面中輸出控件時,產(chǎn)生一個表格(二行二列),以下是表格的樣式 output.AddAttribute(HtmlTextWriterAttribute.Cellspacing, "0"); output.AddAttribute(HtmlTextWriterAttribute.Border, "0"); output.AddAttribute(HtmlTextWriterAttribute.Cellpadding, "0"); output.AddStyleAttribute("LEFT", this.Style["LEFT"]); output.AddStyleAttribute("TOP", this.Style["TOP"]); output.AddStyleAttribute("POSITION", "absolute"); if (Width != Unit.Empty) { output.AddStyleAttribute(HtmlTextWriterStyle.Width, Width.ToString()); } else { output.AddStyleAttribute(HtmlTextWriterStyle.Width, "200px"); } output.RenderBeginTag(HtmlTextWriterTag.Table); //輸出表格 output.RenderBeginTag(HtmlTextWriterTag.Tr); //表格第一行 output.AddAttribute(HtmlTextWriterAttribute.Width, "90%"); output.RenderBeginTag(HtmlTextWriterTag.Td); //以下是第一行第一列中文本框的屬性及其樣式設(shè)置 if (!Enabled) { output.AddAttribute(HtmlTextWriterAttribute.ReadOnly, "true"); } output.AddAttribute(HtmlTextWriterAttribute.Type, "Text"); output.AddAttribute(HtmlTextWriterAttribute.Id, DatePickerInputID); output.AddAttribute(HtmlTextWriterAttribute.Name, DatePickerInputName); output.AddAttribute(HtmlTextWriterAttribute.Value, Text); output.AddStyleAttribute(HtmlTextWriterStyle.Width, "100%"); output.AddStyleAttribute(HtmlTextWriterStyle.Height, "100%"); output.AddStyleAttribute(HtmlTextWriterStyle.FontFamily, Font.Name); output.AddStyleAttribute(HtmlTextWriterStyle.FontSize, Font.Size.ToString()); output.AddStyleAttribute(HtmlTextWriterStyle.FontWeight, Font.Bold?"bold":""); output.AddStyleAttribute(HtmlTextWriterStyle.BackgroundColor, ColorTranslator.ToHtml(BackColor)); output.AddStyleAttribute(HtmlTextWriterStyle.Color, ColorTranslator.ToHtml(ForeColor)); output.RenderBeginTag(HtmlTextWriterTag.Input); //輸出文本框 output.RenderEndTag(); output.RenderEndTag(); output.AddAttribute(HtmlTextWriterAttribute.Width, "*"); output.RenderBeginTag(HtmlTextWriterTag.Td); //以下是第一行第二列中按鈕的屬性及其樣式設(shè)置 if (!Enabled) { output.AddAttribute(HtmlTextWriterAttribute.Disabled, "true"); } output.AddAttribute(HtmlTextWriterAttribute.Type, "Submit"); output.AddAttribute(HtmlTextWriterAttribute.Id, DatePickerButtonID); output.AddAttribute(HtmlTextWriterAttribute.Name, DatePickerButtonName); output.AddAttribute(HtmlTextWriterAttribute.Value, ButtonText); output.AddStyleAttribute(HtmlTextWriterStyle.Width, "100%"); output.AddAttribute(HtmlTextWriterAttribute.Onclick, Page.GetPostBackEventReference(this)); //點擊按鈕時需要回傳服務(wù)器來觸發(fā)后面的OnClick事件 output.AddAttribute(HtmlTextWriterAttribute.Style, ButtonStyle); output.RenderBeginTag(HtmlTextWriterTag.Input); //輸出按鈕 output.RenderEndTag(); output.RenderEndTag(); output.RenderEndTag(); output.RenderBeginTag(HtmlTextWriterTag.Tr); output.AddAttribute(HtmlTextWriterAttribute.Colspan, "2"); output.RenderBeginTag(HtmlTextWriterTag.Td); _Calendar.RenderControl(output); //將日歷子控件輸出 output.RenderEndTag(); output.RenderEndTag(); output.RenderEndTag(); } //復(fù)合控件必須繼承IpostBackEventHandler接口,才能繼承RaisePostBackEvent事件 public void RaisePostBackEvent(string eventArgument) { OnClick(EventArgs.Empty); } protected virtual void OnClick(EventArgs e) { //點擊選擇日期按鈕時,如果日歷子控件沒有顯示則顯示出來并將文本框的值賦值給日歷子控件 if (_Calendar.Attributes["display"] != "") { _Calendar.SelectedDate = DateTime.Parse(Text); _Calendar.Style.Add("display",""); } } //復(fù)合控件中的日歷控件日期變化事件 private void _Calendar_SelectionChanged(object sender, EventArgs e) { //當(dāng)選擇的日期變化時,將所選日期賦值給文本框并將日歷子控件隱藏 Text = _Calendar.SelectedDate.ToString(); _Calendar.Style.Add("display","none"); } } } |
在上面的代碼中,需要注意以下幾點:
·如果你想將此控件的某些屬性供重載,則在聲明屬性前加上virtual關(guān)鍵字;
·在頁面輸出此控件時(即在Render事件中),是先定義子控件的樣式或?qū)傩,然后再產(chǎn)生子控件;
·在隱藏日歷子控件時,建議不要使用Visible屬性來顯示/隱藏,使用Visible=false隱藏時服務(wù)器端將不會將日歷控件HTML代碼發(fā)送給客戶端,會導(dǎo)致復(fù)合控件裝載日歷控件的表格會空白一塊出來,影響頁面的布局。所以使用樣式display=none設(shè)置來使日歷控件在客戶端隱藏,但是HTML代碼依然存在于頁面中;
日歷控件
四、結(jié)束語
在編寫服務(wù)器控件時,需要一定的HTML語言基礎(chǔ),也要清楚.NET程序的請求處理方式。服務(wù)器控件封裝了客戶端行為及邏輯判斷,無需開發(fā)者添加更多代碼。當(dāng)然,有些地方使用服務(wù)器控件可以帶來方便,但是也增加了服務(wù)器的負(fù)荷。有時適當(dāng)?shù)慕Y(jié)合javascript使一些代碼在客戶端運行,可提高WEB應(yīng)用程序效率。