1. Trang chủ
  2. » Công Nghệ Thông Tin

Tài liệu ASP.NET 1.1 Insider Solutions- P2 ppt

50 304 0

Đang tải... (xem toàn văn)

Tài liệu hạn chế xem trước, để xem đầy đủ mời bạn chọn Tải xuống

THÔNG TIN TÀI LIỆU

Thông tin cơ bản

Định dạng
Số trang 50
Dung lượng 1,06 MB

Nội dung

</div> </ItemTemplate> <EditItemTemplate> <asp:Textbox id=”txtPrecis” runat=”server” TextMode=”MultiLine” Rows=”4” Width=”300” Style=”width:300px” Text=’<%# Container.DataItem(“Precis”) %>’ /> </EditItemTemplate> </asp:TemplateColumn> You might like to experiment with the CellPadding and CellSpacing properties of the DataGrid control, as well as with different values of the border style selector, to get a different appearance for the scrollable regions. For example, “style=border:3 inset” when CellPadding and CellSpacing are both zero gives a very compact grid-like effect. Loading Controls Dynamically at Runtime When the ASP team at Microsoft was designing ASP.NET, it probably seemed obvious that the way forward was to compile the pages into some kind of executable code. This approach means that there is a distinct separation between the tasks (and the amount of processing work ASP.NET has to do) of generating a page the first time it is executed—when it has to be compiled and the resulting code written to disk—and subsequent executions of the compiled code. As a result, the way that the structure and content of a page are discovered and created from a file containing declarative definitions and code in <script runat=”server”> sections only affects the “initial hit” performance and not the performance on subsequent requests. Consequently, this has provided a development environment that supports quite complex page creation tech- niques, such as the use of server controls and user controls, page and control state maintenance, and dynamic creation of a control tree for the page. In particular, the use of a developer-accessible control tree has made it really easy to use ASP.NET to build pages that, in ASP 3.0 and many other Web development environments, would required complicated Response.Write statements, #include directives, and other tricks. Being able to create controls dynamically at runtime, meanwhile, is extremely useful if you don’t know beforehand how many instances of a particular control you need on the page. For example, you might need to create a number of text boxes or buttons, depending on the value entered by the user, which could therefore be different each time the page is executed. The ASP.NET Control Tree As ASP.NET processes a page, it generates a control tree that contains references to all the server controls on the page. Note that this only includes server controls—basically declarative elements that contain the runat=”server” attribute. Figure 1.10 shows a conceptual view of a page that 1 Web Forms Tips and Tricks 38 LISTING 1.14 Continued 03 0672326744 CH01 5/4/04 12:22 PM Page 38 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. 39 Loading Controls Dynamically at Runtime contains several server controls, including a server-side <form> element that contains many of the controls on the page. Notice also that the Hyperlink and HtmlAnchor ( <a> element) controls in this example have child Image and HtmlImage controls. These represent the typical output that provides clickable images: <a href=”http://www.daveandal.net”> <img src=”click.gif” border=”0” /> </a> Page HtmlGenericControl Label Image TextBox HtmlInputFile CheckBox Hyperlink Button Image HtmlAnchor HtmlImage HtmlForm Label FIGURE 1.10 A conceptual view of an ASP.NET page that contains nested controls. In more technical terms, the page consists of a hierarchy of control instances. Figure 1.11 shows this in tree form. Each object in the tree is a server control that is descended directly or indi- rectly from System.Web.UI.Control and thus exposes a Controls property that references a ControlCollection instance. Each ControlCollection instance is, as you might guess, a collection of references to all the child controls for that control. You can manipulate the control tree by adding controls to and removing them from these ControlCollection instances. When the page is rendered, the control tree is used to build the HTML (or other output) that is sent to the client. Table 1.1 shows the properties and methods of the ControlCollection object that are useful when manipulating the control tree. 03 0672326744 CH01 5/4/04 12:22 PM Page 39 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. TABLE 1.1 The Members of the ControlCollection Object for Working with, Adding, and Removing Controls Property or Method Description Count Returns the number of controls in the ControlCollection instance. Item Acts as the indexer for the zero-based ControlCollection instance, returning a reference to a control within the collection. Add(control) Adds the Control instance referenced by the control parameter to the end of the ControlCollection instance of this control. AddAt(index, control) Inserts the Control instance referenced by the control parameter into the ControlCollection instance of this control at the specified Integer index . Clear() Removes all the controls from the ControlCollection instance of this control. Contains(control) Returns a Boolean value indicating whether the Control instance referenced in the control parameter is a member of this control’s ControlCollection instance. IndexOf(control) Returns the Integer index of the Control instance referenced in the control parameter within this control’s ControlCollection instance. Remove(control) Removes the Control instance referenced in the control parameter from this control’s ControlCollection instance. RemoveAt(index) Removes the Control instance at the specified Integer index from this control’s ControlCollection instance. Adding Controls to the Control Tree In most cases, you can just use the Add method to add controls to the control tree in the correct order to produce the output you want. This is generally easier than trying to figure out where to insert a control within a collection, although the other methods are useful if you need to do any complex management of the child controls for a particular control. Probably the easiest way to insert a control into a page at a specific point is to use an ASP.NET PlaceHolder control. This generates no output in the page, but it does expose a ControlCollection instance to which you can add other controls. When you use this approach, the newly added 1 Web Forms Tips and Tricks 40 Page Label Label HtmlGeneric Control Image CheckBox TextBox HtmlInputFile Button HtmlAnchor HtmlImage Hyperlink Image HtmlForm FIGURE 1.11 The ASP.NET control tree for the page shown in Figure 1.10. 03 0672326744 CH01 5/4/04 12:22 PM Page 40 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. 41 Loading Controls Dynamically at Runtime controls will always appear in the same position in the page, even if you later add or remove controls from elsewhere in the control tree or the parent control’s ControlCollection instance. The following code demonstrates the use of the PlaceHolder control. In the Page_Load event, you just create a new Hyperlink control and set the NavigateUrl property. Then you create a new Image control and specify its ImageUrl property. Next, you add the Image control to the ControlCollection instance of the Hyperlink control and add the Hyperlink control to the ControlCollection instance of the PlaceHolder control: <asp:PlaceHolder id=”ph1” runat=”server” /> . Sub Page_Load() Dim oLink As New Hyperlink() oLink.NavigateUrl = “http://www.daveandal.net” Dim oImage As New Image() oImage.ImageUrl = “~/images/True.gif” oLink.Controls.Add(oImage) ph1.Controls.Add(oLink) End Sub When the page is rendered, the following output is generated (ASP.NET automatically adds the border=”0” attribute): <a href=”http://www.daveandal.net”> <img src=” /images/True.gif” border=”0” /> </a> Creating a DataGrid Control Dynamically at Runtime Compared to the previous, somewhat trivial example, the following example generates a more complex page which contains a DataGrid control that supports inline editing. The result is shown in Figure 1.12. The Actual Output Format The output is not actually indented as shown here, but instead is generated as a single line with no spaces or carriage returns. In this example it is formatted with carriage returns and indented so that you can see the result more clearly. FIGURE 1.12 A dynamically generated DataGrid control. 03 0672326744 CH01 5/4/04 12:22 PM Page 41 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. The page contains all the server-side code to handle the events in the DataGrid control, using the same techniques in the first example in this chapter. (We won’t be looking at that code here.) With this example, we are interested in the way that the DataGrid control itself is generated. The HTML section of the sample page contains a server-side <form> element, but no other content: <form id=”frmMain” runat=”server”> <!-- DataGrid and Label will be dynamically inserted here --> </form> Instead, the DataGrid control and the Label control used to display any data access errors are added to the page dynamically during the Page_Load event. Also, the various events in the DataGrid control are wired to the appropri- ate event handlers already located in the <script> section of the page. The previous example uses a PlaceHolder control as the container to which the new controls are added. However, a server-side <form> element works just as well, and in this example you can add the controls directly to the ControlCollection instance of the HtmlForm control that implements the server-side <form> control. Setting Size and Color Properties Dynamically You can generate values for some of the properties of Web Forms server controls. Properties that set the color of parts of the output, such as ForeColor and BackColor , accept references to a Color structure. Properties that accept sizes, such as Width and BorderWidth , accept references to a Unit structure. When declaring a server control in the HTML section of the page, you can use the color names or size values directly, as in this example: <HeaderStyle ForeColor=”#ffffff” BackColor=”#b50055” /> <ItemStyle BorderWidth=”1px” /> However, to set these properties dynamically, you have to provide an instance of the correct structure classes. Here’s an example: oGrid.HeaderStyle.ForeColor = Color.FromName(“#ffffff”) oGrid.HeaderStyle.BackColor = Color.FromName(“#b50055”) oGrid.ItemStyle.BorderWidth = Unit.Pixel(1) The Unit structure is part of the System.Web.UI.WebControls namespace, so it is available by default in all ASP.NET Web pages. However, the Color structure is defined in the System.Drawing 1 Web Forms Tips and Tricks 42 Choosing the Event when Adding Controls Instead of generating the controls during other ASP.NET page events, such as Init or Render, we had most success getting the process to work reliably, especially when wiring up event handlers, by using the Page_Load event. The controls must be generated on every postback (not just when the page is first loaded) and in exactly the same order and with the same ID values. Unlike control values, dynamically generated controls are not maintained in the viewstate of the page. However, values are maintained and will be reloaded after the controls have been created and added to the control tree. 03 0672326744 CH01 5/4/04 12:22 PM Page 42 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. 43 Loading Controls Dynamically at Runtime namespace, which is not imported into ASP.NET pages by default. Therefore, you have to include the appropriate Import directive in any pages that reference a Color structure: <%@Import Namespace=”System.Drawing” %> Creating the DataGrid Control Creating the DataGrid control itself is not difficult; it just requires quite a lot of repetitive code. You create an instance of a DataGrid control, set all the properties, and then you add it to the Controls collection of the <form> element in the page. You do the same with the Label control that will display any data access errors. Listing 1.15 shows the declaration of two page-level variables that are used to hold references to the new controls (so that they can be accessed in routines other than the Page_Load event handler), followed by the start of the Page_Load event handler. Here, you create the DataGrid control and add all the properties that set the appearance and behavior of the control. You can reduce the amount of code required by taking advantage of the Visual Basic .NET With construct. Notice how you set the properties of objects that are actually children of the DataGrid control, such as the HeaderStyle , ItemStyle , and AlternatingItemStyle objects. You can use a nested With construct or just reference them by using a period to access the child objects. LISTING 1.15 Dynamically Generating a DataGrid Control Dim oGrid As DataGrid Dim oLabel As Label Sub Page_Load() ‘ create a DataGrid control oGrid = New DataGrid() ‘ set control properties With oGrid .id = “dgr1” .BorderStyle = BorderStyle.None .BorderWidth = Unit.Pixel(0) .BackColor = Color.FromName(“#deba84”) .CellPadding = 3 .CellSpacing = 0 .DataKeyField = “CustomerID” .Width = Unit.Percentage(100) .AutoGenerateColumns = False With .HeaderStyle .Font.Bold = True .ForeColor = Color.FromName(“#ffffff”) .BackColor = Color.FromName(“#b50055”) End With 03 0672326744 CH01 5/4/04 12:22 PM Page 43 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. .ItemStyle.BackColor = Color.FromName(“#fff7e7”) .AlternatingItemStyle.BackColor = Color.FromName(“#ffffc0”) End With ‘ create a column for the DataGrid control ‘ and set properties Dim oCol1 As New EditCommandColumn() With oCol1 .EditText = “Edit” .CancelText = “Cancel” .UpdateText = “Update” End With ‘ add column to DataGrid oGrid.Columns.Add(oCol1) ‘ repeat for remaining columns Dim oCol2 = New BoundColumn() With oCol2 .DataField = “CustomerID” .HeaderText = “ID” .ReadOnly = True End With oGrid.Columns.Add(oCol2) . ‘ same for CompanyName, City, Country and Phone columns . Toward the end of Listing 1.15, you can see the columns being added. You create an instance of the appropriate type of column, set the properties, and then add the column to the Columns collection. Like the Controls property, the Columns property of a DataGrid control is a collection of references to the columns that make up the DataGrid control. Listing 1.15 does not contain the declarations of the Company Name, City, Country, and Phone columns because they are identical to the Customer ID column (except, of course, that they refer to different columns in the source data). Wiring Up the DataGrid Control Events With the DataGrid control complete, you can attach its events to the appropriate event handler routines already present in the page, as shown in Listing 1.16. In Visual Basic .NET you use the AddHandler statement, and in C# you just append the event delegates by using the += operator. You can wire up the EditCommand , UpdateCommand , and CancelCommand events, targeting them at the 1 Web Forms Tips and Tricks 44 LISTING 1.15 Continued 03 0672326744 CH01 5/4/04 12:22 PM Page 44 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. 45 Loading Controls Dynamically at Runtime event handlers named DoItemEdit , DoItemUpdate , and DoItemCancel . This achieves the same result as declaring them directly in the page, as in the DataGrid control example earlier in this chapter: OnEditCommand=”DoItemEdit” OnUpdateCommand=”DoItemUpdate” OnCancelCommand=”DoItemCancel” LISTING 1.16 Wiring Up the Event Handlers, Creating the Label Control, and Populating the DataGrid Control . ‘ add event handlers to the grid AddHandler oGrid.EditCommand, _ New DataGridCommandEventHandler(AddressOf DoItemEdit) AddHandler oGrid.UpdateCommand, _ New DataGridCommandEventHandler(AddressOf DoItemUpdate) AddHandler oGrid.CancelCommand, _ New DataGridCommandEventHandler(AddressOf DoItemCancel) ‘ create new Label control and set properties oLabel = New Label() With oLabel .id = “lblErr” .EnableViewState = False End With ‘ add new controls to page as children of <form> frmMain.Controls.Add(oGrid) frmMain.Controls.Add(oLabel) ‘ only need to databind if it is not a postback ‘ viewstate used to populate dynamically added controls If Not Page.IsPostback Then oGrid.DataSource = GetCustomers() oGrid.DataBind() End If End Sub After attaching the event handlers, you generate a new Label control and set its properties. Then you add the DataGrid and the Label controls to the ControlCollection instance of the server-side <form> control declared in the page (as shown in Listing 1.16). Populating the DataGrid Control The final task in this example, shown at the end of Listing 1.16, is to populate the DataGrid control. As long as the control tree you generate is the same every time the page is loaded, the 03 0672326744 CH01 5/4/04 12:22 PM Page 45 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. values of all the controls will be maintained through the viewstate of the page—even for dynamically added controls. So you only have to perform the data binding to the data source if this is not a postback, just as you would if you had declared the DataGrid control directly within the HTML section of the page. The viewstate of the page also stores the values of many of the other properties of the controls on the page. So if you allow users to modify properties, such as whether specific columns are visible or the color of the text, you’ll want these values to be preserved across page loads and not be reset every time you regenerate the control. In this case, you can set the values only the first time the page loads, at the same time as populating the DataGrid control. For example, this code sets the style of the header row only when the page first loads, but it is maintained across postbacks within the viewstate of the page: If Not Page.IsPostback Then oGrid.DataSource = GetCustomers() oGrid.DataBind() oGrid.HeaderStyle.Font.Bold = True oGrid.HeaderStyle.ForeColor = Color.FromName(“#ffffff”) oGrid.HeaderStyle.BackColor = Color.FromName(“#b50055”) End If Loading User Controls Dynamically at Runtime The final topic we’ll briefly look at to finish this chapter is dynamically loading user controls at runtime. In theory, the principles are the same as for the DataGrid control; however, there are a couple things to be aware of with user controls. A user control is not strongly typed—in other words, it is usually generated as an instance of the generic UserControl class, whereas other server controls are specific classes from the .NET Framework class library. You can use the LoadControl method of the Page object to load a user control. The following code takes the path and name of the .ascx disk file and returns a reference to the control as a UserControl instance that you can add to the ControlCollection instance of any other control: Dim oNewCtrl As UserControl = LoadControl(“path-to-ascx-file”) oExistingControl.Controls.Add(oNewCtrl) This is fine if the user control is simply some static user interface content. However, if you want to access properties or other members of the user control, you have a problem because the UserControl class that represents the user control doesn’t expose them. In that case, you have to add to the page a reference to the user control, and you have to specify the classname of the user control in the .ascx file. When you insert a user control in the page declaratively, you use a Register directive to specify the tag prefix and tag name you’ll be using, and you use the path and name of the .ascx file that implements the user control. Here’s an example: <%@Register TagPrefix=”ahh” TagName=”Spinbox” Src=” \ascx\user-spinbox.ascx” %> 1 Web Forms Tips and Tricks 46 03 0672326744 CH01 5/4/04 12:22 PM Page 46 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. 47 Loading Controls Dynamically at Runtime When you want to insert a user control dynamically and be able to access it as a strongly typed object, you use the Reference directive instead. The following example just takes the path and name of the user control: <%@Reference Control=” \ascx\user-spinbox.ascx” %> However, this assumes that the user control itself declares a classname. In the user control, you have to add the ClassName attribute to the Control directive, as in this example: <%@Control Language=”VB” ClassName=”UserSpinBox” %> Now you can use the CType statement in Visual Basic (or a direct cast in C#) to convert the UserControl reference into a reference to the specific class. For example, you can load an instance of the SpinBox user control you’ll be meeting later in this book and expose it as a UserSpinBox instance with the following code: Dim oCtrl As UserControl = LoadControl(“ \ascx\user-spinbox.ascx”) Dim oSpinBox As UserSpinBox = CType(oCtrl, UserSpinBox) Alternatively, if you just want to set a property (such as the Increment property), you can use something like this: Dim oCtrl As UserControl = LoadControl(“ \ascx\user-spinbox.ascx”) CType(oCtrl, UserSpinBox).Increment = 3 oPlaceHolder.Controls.Add(oCtrl) An Example of Loading a User Control To briefly demonstrate the dynamic loading of a user control, the final example in this chapter loads instances of the custom SpinBox user control, as shown in Figure 1.13. The page contains both a Register and a Reference directive for the SpinBox user control: <%@Register TagPrefix=”ahh” TagName=”Spinbox” Src=” \spinbox\ascx\user-spinbox.ascx” %> <%@Reference Control=” \spinbox\ascx\user-spinbox.ascx” %> The first SpinBox instance you see in the page is inserted declaratively, which is possible because of the presence of the Register direc- tive. However, as you can see from Listing 1.17, the remainder of the page is made up basically of three PlaceHolder controls where you can dynamically add the other instances of the SpinBox control. Running the SpinBox Example As you’ll see in Chapter 8, “Building Adaptive Controls,” which discusses the SpinBox control, you have to copy a file that we provide with the samples into the aspnet_client folder of your Web site for this example to work. You should copy the file spinbox.js from the samples into a new subfolder named custom within the aspnet_client folder of your Web site. 03 0672326744 CH01 5/4/04 12:22 PM Page 47 Please purchase PDF Split-Merge on www.verypdf.com to remove this watermark. [...]... example by clicking the Submit (Normal) button in the original page, you see an ASP.NET error page, indicating that the viewstate for the page is corrupted (see Figure 2.2) This is because ASP.NET encodes the viewstate it stores in the page along with the control tree and other details of the original page When a postback occurs, ASP.NET validates this encoded data against the current page to act as a guard... the HttpContext object that ASP.NET uses to keep track of the original page If you check out the properties of the HttpContext class, you’ll see that it provides access to everything that you use in your ASP.NET code You can access the current Request, Response, Server, Session, and Application objects, plus objects such as User (details of the current user), Cache (the ASP.NET global user cache),... chapter demonstrates how you can achieve it Techniques for Passing Values Between Pages ASP.NET engenders a postback architecture, where pages containing a server-side form (a element that contains the runat=”server” attribute) are always posted back to themselves In fact, this is effectively enforced by ASP.NET, which doesn’t allow server-side code to set the action attribute of a server-side... controls on the form from the Request collections in the traditional way This also allows you to have multiple forms on the page, but it prevents you from using many of the ASP.NET server controls on the form It also prevents the ASP.NET postback architecture from working, so you cannot access the controls on the original page—you can access only their posted values n Use client-side script to change... { var theForm = document.getElementById(‘frmMain’); theForm.action = sURL; } In the case of the first button, the target page is catchrequest.aspx This page contains an ASP.NET Label control and a server-side element with an ASP.NET Button control: 53 54 2 Cross-Page Posting Values in the Request collections: . Hyperlink Image HtmlForm FIGURE 1. 11 The ASP. NET control tree for the page shown in Figure 1. 10. 03 0672326744 CH 01 5/4/04 12 :22 PM Page 40 Please purchase. attribute. Figure 1. 10 shows a conceptual view of a page that 1 Web Forms Tips and Tricks 38 LISTING 1. 14 Continued 03 0672326744 CH 01 5/4/04 12 :22 PM Page

Ngày đăng: 24/12/2013, 04:16