ObjectDataSource is really a fun feature, in addition to being very useful. Instead of getting the data from any type of data storage, as would be the case with other DataSource controls, it simply uses business objects (classes that just store your application data). This is even more interesting if you work with the new Object-Spaces feature set included with Microsoft Visual Studio Whidbey Mobile Control through which objects are stored directly in the database.
So what do you need first of all to display business objects? Yes, some business objects! I've already created two for holding personal information in Listing 3-8: the Person class, with the Id, Firstname, and Lastname properties; and the PersonCollection class, which is based on Generics.
// Person.cs using System; public class Person { private int id; private string firstname; private string lastname; public Person(int id, string firstname, string lastname) { this.id = id; this.firstname = firstname; this.lastname = lastname; } public int Id { get { return this.id; } set { this.id = value; } } public string Firstname { get { return this.firstname; } set { this.firstname = value; } } public string Lastname { get { return this.lastname; } set { this.lastname = value; } } } // PersonCollection.cs using System; using System.Collections.Generic; public class PersonCollection : List<Person> { public void Remove(int id) { Person person = this.FindPersonById(id); if (person != null) { base.Remove(person); } } public Person FindPersonById(int id) { foreach (Person person in this) { if (person.Id.Equals(id)) { return person; } } return null; } }
The ObjectDataSource control expects to receive a qualified name of a class, which is responsible for processing the objects. Usually this is a class of your business layer. The first step of the implementation of a PersonManager class is shown in Listing 3-9. The SelectPersons() method creates a new PersonCollection, fills it with data, and returns it afterwards. The data is also stored in application scope. On consecutive calls, the data will be read from there.
using System; using System.Web; public class PersonManager { private const string personsKey = "persons"; public PersonCollection SelectPersons() { HttpContext context = HttpContext.Current; if (context.Application[personsKey] == null) { PersonCollection persons = new PersonCollection(); persons.Add(new Person(0, "Patrick", "Lorenz")); persons.Add(new Person(1, "Micha", "Brunnhuber")); persons.Add(new Person(2, "Thomas", "Ballmeier")); persons.Add(new Person(3, "Marc", "Höppner")); context.Application[personsKey] = persons; } return (context.Application[personsKey] as PersonCollection); } }
OK, that's enough preparation. Now let's get to the meat. Add a GridView control to a new page and create a new ObjectDataSource control by using the wizard. Modify the properties of the new source and assign PersonManager to TypeName and SelectPersons to SelectMethod. Ready, steady, go! Hit the F5 key and the result should look like what you see in Figure 3-22.
Tip |
Of course, the data source, or rather the GridView control, allows you to define explicitly the properties that will be visualized including their order. Just disable the automatic column generation and define them manually, as usual. |
The use of the ObjectDataSource control combined with parameters is pretty interesting. You can assign them to the data source and specify the data type to use as shown earlier and demonstrated in Figure 3-23. Here, I'll talk about SelectParameters.
The defined parameters are passed to the select method. Unique names are required, and the data types must match. Overloading isn't supported. Be careful in the current Alpha version, which is case sensitive (this may change in the final release).
For this example, I have written a new method called SelectPerson (singular) which expects an ID as parameter (see Listing 3-10). The returned object is of type Person and gets bound to a DetailsView control. The ID is entered into a TextBox control, which serves as the source for the parameter. Figure 3-24 shows how to use it.
// PersonManager.cs using System; using System.Web; public class PersonManager { private const string personsKey = "persons"; public PersonCollection SelectPersons() { HttpContext context = HttpContext.Current; if (context.Application[personsKey] == null) { PersonCollection persons = new PersonCollection(); persons.Add(new Person(0, "Patrick", "Lorenz")); persons.Add(new Person(1, "Micha", "Brunnhuber")); persons.Add(new Person(2, "Thomas", "Ballmeier")); persons.Add(new Person(3, "Marc", "Höppner")); context.Application[personsKey] = persons; } return (context.Application[personsKey] as PersonCollection); } public Person SelectPerson(int id) { return this.SelectPersons().FindPersonById(id); } } // ObjectDataSource2.aspx <%@ page language="C#" %> <html> <head runat="server"> <title>Untitled Page</title> </head> <body> <form runat="server"> ID: <asp:textbox id="TextBox1" runat="server" width="32px">0</asp:textbox> <asp:button id="Button1" runat="server" text="Show Person" /> <br /> <br /> <asp:detailsview id="DetailsView1" runat="server" datasourceid="ObjectDataSource1"> </asp:detailsview> <asp:objectdatasource id="ObjectDataSource1" runat="server" typename="PersonManager" selectmethod="SelectPerson"> <selectparameters> <asp:controlparameter name="id" propertyname="Text" type="Int32" controlid="TextBox1"> </asp:controlparameter> </selectparameters> </asp:objectdatasource> </form> </body> </html>
The features of the ObjectDataSource control don't stop at the visualization of data. It also provides editing and even deleting capabilities, if your business object allows this (see Figure 3-25). All you have to do is to add two more methods, Update and DeletePerson, as shown in Listing 3-11, and you're almost done. To get this example running, you have to assign the additional methods to the ObjectDataSource control and set the DataKeyNames property of the GridView control to Id.
// PersonManager.cs using System; using System.Web; public class PersonManager { private const string personsKey = "persons"; ... public void DeletePerson(int Id) { HttpContext context = HttpContext.Current; PersonCollection persons = (context.Application[personsKey] as PersonCollection); persons.Remove(Id); } } // ObjectDataSource3.aspx <%@ page language="C#" %> <html> <head id="Head1" runat="server"> <title>Untitled Page</title> </head> <body> <form id="Form1" runat="server"> <asp:gridview id="GridView1" runat="server" datasourceid="ObjectDataSource1" autogeneratecolumns="False" datakeynames="Id"> <columnfields> <asp:boundfield datafield="Id" readonly="True" headertext="ID"> </asp:boundfield> <asp:boundfield datafield="Firstname" headertext="Firstname"> </asp:boundfield> <asp:boundfield datafield="Lastname" headertext="Lastname"> </asp:boundfield> <asp:commandfield showeditbutton="True"> </asp:commandfield> <asp:commandfield showdeletebutton="True"> </asp:commandfield> </columnfields> </asp:gridview> <asp:objectdatasource id="ObjectDataSource1" runat="server" typename="PersonManager" selectmethod="SelectPersons" updatemethod="Update" deletemethod="DeletePerson"> </asp:objectdatasource> </form> </body> </html>
Last but not least, I should show you how to insert new data records. One solution is to use a combination of a GridView control and a DetailsView control as discussed previously. Because all parameters are returned as strings by the input fields, it's crucial to specify the adequate data types while defining the parameters.
Only a few changes are necessary. You need an additional DetailsView control, which you have to bind to the existing data source. The rest is done by the Insert method in the PersonManager class (see Listing 3-12). Inserting the record for a new person is shown in Figure 3-26.
// PersonManager.cs using System; using System.Web; public class PersonManager { private const string personsKey = "persons"; ... public void Insert(int Id, string Firstname, string Lastname) { HttpContext context = HttpContext.Current; PersonCollection persons = (context.Application[personsKey] as PersonCollection); persons.Add(new Person(Id, Firstname, Lastname)); } } // ObjectDataSource4.aspx ... <asp:objectdatasource id="ObjectDataSource1" runat="server" typename="PersonManager" selectmethod="SelectPersons" deletemethod="DeletePerson" updatemethod="Update" insertmethod="Insert"> <insertparameters> <asp:parameter name="Id" type="Int32"></asp:parameter> </insertparameters> </asp:objectdatasource> <asp:detailsview id="DetailsView1" runat="server" datasourceid="ObjectDataSource1" defaultmode="Insert" autogeneraterows="False" datakeynames="Id"> <rowfields> <asp:boundfield datafield="Id" headertext="ID:"> </asp:boundfield> <asp:boundfield datafield="Firstname" headertext="Fn:"> </asp:boundfield> <asp:boundfield datafield="Lastname" headertext="Ln:"> </asp:boundfield> <asp:commandfield showinsertbutton="True"> </asp:commandfield> </rowfields> </asp:detailsview> </form> </body> </html>