|
Posted on 2007-03-08 22:05 玄铁剑 阅读(338) 评论(0) 编辑 收藏 引用 所属分类: asp.net
Introduction
The ASP.NET 2.0 GridView is a wonderful beast that can allow you to create new database rows, edit in place, even provide DropDownList s and CheckBox es to aid you in the entry process. It can even do data validation.
I have added the following additional functionality over and above what you get "out of the box":
- Insert new rows.
- Edit via
DropDownList s.
- Validation.
- Delete confirmation.
- Checkbox input.
- Current row highlighting.
- Selective cell highlighting based on value.
None of the functionality that I will show you how to build is rocket science, and most of it is around the internet someplace (although I have never seen the "Insert new row before"). This is my humble attempt to consolidate it all in one article.
Background
I was presented with a project to convert an Access single-user application into a Web-based multi-user SQL Server application. The Access application had about 10 tables that needed to be managed through an Add/Edit/Delete interface. This table maintenance was to be done by a single individual, so concurrency issues would not be a problem. I decided that with some tweaking, the GridView could perform what needed to be done. Out of the box, the GridView will not do row inserts, nor will it provide DropDownList s where the data to be edited is really an ID representing a row in another table. This example uses the NorthWind Products table to add and update data rows. It is a step-by-step guide to producing a web page which will accomplish this task. It is long and arduous, but this is what it takes. When you have completed this exercise, you should be ready to tackle your own tables with confidence. You may want to print out the steps and keep them handy so you don't have to be constantly switching screens (unless you have two monitors ;-)
Using the code
Download the project, unzip it to somewhere on your hard disk, and then create a new web project from an existing folder. Build the project and test drive it so that you see what it does. Now, create a new ASPX file, Default2.aspx, and do it yourself manually.
Steps to create the project manually
- Create a blank project.
- Create an Images folder.
- Right click on Images.
- Choose "Add Existing Item".
- Go to the folder you unzipped the download to, and select all the .GIF files and click "Add".
- In the design view, add a
DataSource .
- Configure the
DataSource to select each individual field.
- Create a
GridView and select the previously created DataSource .
- Click the
GridView SmartTag to open up its Tasklist.
- Enable Paging, Sorting, Editing and Deleting.
- Select Edit Columns.
- In the lower right pane, the top item is
CommandField .
- Click it, and the top right pane shows the attributes.
- Change Appearance > Button Type to
Image .
- Set
CancelButtonUrl to Images/Cancel.gif.
- Set the rest of the
ButtonUrl s appropriately.
- Select ProductID from the lower left pane.
- Press the red X to the right to remove it as it is not an editable field.
- Build and run the project.
You now have a working editable/delete-able table maintenance program.
Editing using DropDownLists
- Create a new
SqlDataSouce (in the Toolbox under Data).
- Configure this
DataSource to connect to the Suppliers table, and in the SELECT statement, choose SupplierID and CompanyName .
- Create a new
SqlDataSouce (you know where it is).
- Configure this
DataSource to connect the the Categories table, and in the SELECT statement, choose CategoryID and CategoryName .
- Click on the
GridView SmartTag to bring up the Tasklist.
- Click "Edit Columns...".
- Highlight SupplierID (in the lower left pane), then click on the phrase "Convert this field into a template" (over to the right).
- Do the same for CategoryID.
- Click on OK.
- You should now be looking at the
GridView Tasklist.
- Click Edit Templates at the bottom.
- Click the down arrow and you will see a list of templates, choose (under Column [2] - SupplierID) "
EditItemTemplate ".
- Do not choose "
ItemTemplate ".
- Select the text box and delete it.
- From the toolbox, drag a
DropDownList control and place it where the textbox was.
- Click its SmartTag and click "Choose data source".
- Choose the Supplier
DataSource that you set up earlier.
- Bind the display field to
CompanyName , and the value to SupplierID .
- Click the
GridViews SmartTag.
- Now, pick "
EditItemTemplate ", but this time, under Column [3] - CategoryID.
- Repeat what you have just done to connect up the
CategoryID .
- Now, build and run the application.
- You will see in the
SupplierID and CategoryID columns, the Supplier/Category IDs (we'll fix that shortly), but when you click Edit, they magically transform into dropdown lists (well, it's not really magic, but it is pretty impressive).
- Click Cancel (because we're not ready to do an Update yet).
- Close the application.
- Now, we're going to turn those irritating Supplier/Category IDs into more user friendly names.
- You need to modify the first
DataSource you created to dig the supplier's name and the category's name out of their respective tables. You could do it the manly (stupid) way by editing the source code for the SELECT statement and doing the INNERJOIN s yourself, or just click the DataSource 's SmartTag and click "Configure data source".
- Click Next once, and then change the retrieval type to Custom SQL Statement.
- Click Next and make sure Select is the highlighted tab.
- Click Query Builder.
- Right click in the pane showing the Products table and choose Add table.
- Add the following tables: Suppliers, Categories, then close.
- Add CompanyName from Suppliers and CategoryName from Categories to the query result.
- You're now done with the
DataSource . Now, click the GridView SmartTag. At some point, an alert box will ask you if you want to clear the work you have already done. Just say no!
- Click Edit Templates in the Tasklist.
- Choose the ItemTemplate for Column [2] - SupplierID.
- Click the SmartTag for the
Label control and choose Edit Databindings.
- Select Text in the left pane and change the Bound to column to
CompanyName .
Important: Make sure that the two-way binding is still checked after you do this.
- Close the box.
- Click the SmartTag for GridView1 - Column [2] - SupplierID.
- Choose the
ItemTemplate for Column [3] - CategoryID.
- Click the SmartTag for the
Label control and choose Edit Databindings.
- Select Text in the left pane and change the Bound to column to
CategoryName .
Important: Make sure that the two-way binding is still checked after you do this.
- Close the box.
- Click the SmartTag for
GridView1 - Column [3] - CategoryID.
- Click End Template Editing.
- Build and execute the application.
Now, you will see that CategoryID and SupplierID numbers have been replaced by names.
- Click around and edit the various lines but don't save - this is the real live NorthWind database that other tutorials depend on. We'll get to adding next, so you can add your own rows and mangle/delete them without injuring the real data.
Follow these steps to make the GridView - Add New Records
- Click the Product DataSource SmartTag.
- Click Configure data source.
- Click Next twice.
- Make sure Select is the highlighted tab.
- Click Query Builder.
- Change the existing statement:
SELECT Products.ProductID, Products.ProductName, Products.SupplierID,
Products.CategoryID, Products.QuantityPerUnit, Products.UnitPrice,
Products.UnitsInStock, Products.UnitsOnOrder, Products.ReorderLevel,
Products.Discontinued, Categories.CategoryName, Suppliers.CompanyName
FROM Products INNERJOIN Categories ON
Products.CategoryID = Categories.CategoryID
INNERJOIN Suppliers ON
Products.SupplierID = Suppliers.SupplierID
ORDERBY Products.ProductName to be: Collapse SELECT0 as ProductID, '' as ProductName, 0 as SupplierID, 0 as
CategoryID, '' as QuantityPerUnit, 0.00 as UnitPrice,
0 as UnitsInStock, 0 as UnitsOnOrder, 0 ReorderLevel,
convert(bit, 0) as Discontinued,'' as CategoryName,
'' as CompanyName
UNIONSELECT Products.ProductID, Products.ProductName, Products.SupplierID,
Products.CategoryID, Products.QuantityPerUnit, Products.UnitPrice,
Products.UnitsInStock, Products.UnitsOnOrder, Products.ReorderLevel,
Products.Discontinued, Categories.CategoryName, Suppliers.CompanyName
FROM Products INNERJOIN Categories ON
Products.CategoryID = Categories.CategoryID INNERJOIN
Suppliers ON Products.SupplierID = Suppliers.SupplierID
ORDERBY Products.ProductName
- Insert the following JavaScript code into the
<head> of the ASPX file:
Collapse <script>
function FixGrid(idGrid, PageIndex, EditIndex)
{
var Start = 1;
if(EditIndex != 0)
{
if(PageIndex == 0)
{
Start = 2;
idGrid.firstChild.childNodes[1].childNodes[0].childNodes[0].src=
"Images/Add.gif";
var i =
idGrid.firstChild.childNodes[1].
childNodes[0].innerHTML.indexOf(" ");
idGrid.firstChild.childNodes[1].childNodes[0].innerHTML =
idGrid.firstChild.childNodes[1].childNodes[0].innerHTML.slice(0, i);
}
}
for(var i=Start; ; i++)
{
try
{
var ctl=idGrid.firstChild.childNodes[i].childNodes[0].childNodes[2];
if(ctl.tagName == "INPUT")
{
var onc = ctl.onclick.toString();
if(onc.indexOf("Delete$") == -1)
continue; var j = onc.indexOf("__do");
var k = onc.indexOf(")", j)+1;
onc = "if(confirm('Are you sure') == false)" +
" return(false); "+onc.slice(j, k);
ctl.onclick = onc;
ctl.outerHTML = ctl.outerHTML;
}
}
catch(e)
{
break;
}
}
}
</script>
- Put this script after the closing
</form> tag: <script>
FixGrid(document.all.GridView1,
<%=GridView1.PageIndex.ToString()%>,
<%=EditIndex %>);
</script>
- Right click the
GridView and select Properties.
- Click the Event button.
- Double click Action > RowUpdating.
- Double click Action > RowEditing.
- This will create the empty event handlers and the required event hookup code.
- Now, replace the code in the empty ASPX.CS file with:
Collapse public partial classYOUR_CLASSNAME_HERE : System.Web.UI.Page
{
publicint EditIndex = -1;
protectedvoid Page_Load(object sender, EventArgs e)
{
}
protectedvoid GridView1_RowUpdating(object sender,
GridViewUpdateEventArgs e)
{
if (e.RowIndex > 0 || GridView1.PageIndex > 0)
return;
System.Web.UI.WebControls.SqlDataSource ds =
(System.Web.UI.WebControls.SqlDataSource)
this.FindControl(this.GridView1.DataSourceID);
System.Data.SqlClient.SqlConnection conn =
new System.Data.SqlClient.SqlConnection(ds.ConnectionString);
conn.Open();
string s = ds.InsertCommand;
System.Data.SqlClient.SqlCommand c =
new System.Data.SqlClient.SqlCommand(s, conn);
System.Data.SqlClient.SqlParameter p;
foreach (System.Collections.DictionaryEntry x in e.NewValues)
{
p = new System.Data.SqlClient.SqlParameter("@" + x.Key, x.Value);
c.Parameters.Add(p);
}
c.ExecuteNonQuery();
}
protectedvoid GridView1_RowEditing(object sender,
GridViewEditEventArgs e)
{
EditIndex = e.NewEditIndex;
}
}
-
Now the hack:
The INSERT statement that is generated by VS2005 can not be executed as is. It is originally: InsertCommand="INSERTINTO [Products] ([ProductName],
[SupplierID], [CategoryID], [QuantityPerUnit], [UnitPrice],
[UnitsInStock], [UnitsOnOrder], [ReorderLevel], [Discontinued])
VALUES (@ProductName, @SupplierID, @CategoryID, @QuantityPerUnit,
@UnitPrice, @UnitsInStock, @UnitsOnOrder, @ReorderLevel, @Discontinued)" You need to change it to: InsertCommand="INSERTINTO [Products] ([ProductName], [SupplierID],
[CategoryID], [QuantityPerUnit], [UnitPrice], [UnitsInStock],
[UnitsOnOrder], [ReorderLevel], [Discontinued])
VALUES (@ProductName, @SupplierID, @CategoryID,
@QuantityPerUnit, convert(money, @UnitPrice),
@UnitsInStock, @UnitsOnOrder, @ReorderLevel, @Discontinued)" You can change it manually in the source code or through the IDE.
Checkboxes
You pretty much get checkboxes for free. The GridView by default will bind a SQL column of type bit (and only bit ) to a CheckBoxField (you'll get a runtime conversion exception if you don't). If you look at the SelectCommand that we modified above, I had to add convert(bit, 0) as Discontinued .
Sorting
By default, the SortExpression for a field is the same as the field. In this case, we are displaying CompanyName and CategoryName in the SupplierID and SortExpression fields, so I have set the SortExpression for SupplierID to CompanyName and the SortExpression for CategoryID to CategoryName .
Look in the code behind file, and you will see how sort direction glyphs were added (code courtesy of Dino Esposito's New Grid in Town article). This code makes use of the OnRowCreated event, so make sure you hook it up in the GridView 's Property Events sheet.
Other business
Data presentation
There are two ways you can format data.
To statically format the data
- Choose Edit Column from the
GridView 's SmartTag.
- Highlight the column you want to format in the lower left pane. Scroll the right hand pane to the bottom.
- Expand
ItemStyle and make your changes.
- In this sample program, I have set each item's
VerticalAlign to top , you'll understand why in the section on validation.
- I also set the
HorizontalAlign to right for the numeric columns.
- If the item you have chosen is also an
ItemTemplate , you can also make some formatting changes to its constituent controls by navigating to it as described above and making your required changes to the ItemTemplate not the EditItemTemplate .
To format the data at runtime
You must create a RowDataBound event handler. To do this:
- Right click the
GridView and select Properties.
- Click the lightning bolt at the top of the Properties window to access the events sheet.
- Double click in the input area to the right of
RowDataBound to auto-generate an empty event handler.
I then modified the empty template to look like this:
Collapse
protected
void GridView1_RowDataBound(object sender,
GridViewRowEventArgs e)
{
if (e.Row.RowType == DataControlRowType.DataRow)
{
if (e.Row.RowIndex != EditIndex)
{
int unitsInStock =
Convert.ToInt32(System.Web.UI.DataBinder.Eval(e.Row.DataItem,
"UnitsInStock"));
if (unitsInStock == 0)
e.Row.Cells[7].BackColor = System.Drawing.Color.Yellow;
e.Row.Attributes.Add("onmouseover",
"this.originalcolor=this.style.backgroundColor;" +
" this.style.backgroundColor='Silver';");
e.Row.Attributes.Add("onmouseout",
"this.style.backgroundColor=this.originalcolor;");
}
}
}
The first part of the above code gets the value of the UnitsInStock field from the current row, and if the value is 0, change the cell's background to yellow. If you have seen other examples of RowDataBound functions, you may wonder why not just do Convert.ToInt32(e.Row.Cells[7].Text) . Well here is the nasty secret that no-one wants to tell you about (or at least not document anywhere). Once you turn a BoundField into an ItemTemplate , you must access the original data, because at the time that RowDataBound is called, it is not yet in the cell for you to access. But if you think about it, this makes perfect sense!
At the RowDataBound time, the cell still contains the controls which make up the template, and there may be good reasons for you to still want to manipulate the constituent controls before rendering.
The second part shows you how to highlight the row that the mouse is over. The only trick here is to save the underlying background color (in the custom attribute originalcolor ) before changing it so that it can be restored on the OnMouseOut event. I have seen other solutions that hard code the background color in the restore, but that technique does not handle alternating background colors.
Validation
I have added validation controls to each EditItemTemplate . Each control is validated as it is edited (i.e.: when it loses focus). If there is an error, the error message is displayed below the control. The reason why I formatted each control to be VerticalAlign = top was so that the line would look nicer when there were errors.
Styles
All elements, whether DataBound , ItemTemplate s, or controls within an ItemTemplate , expose a CssClass element to which you can assign one or more classes to affect the look of the element. If you are specifying more than one class, separate them with a space. It is interesting that none of the GridView elements support inline styling via the CssStyle.Add(...) method that most other classes support. This means that you can not add individual inline style elements to a cell.
Images
If anyone has better images that I can use for this example, please send them to me or send the URLs. These ones suck.
Conclusions
|