Monday, October 10, 2011
AJAX DropDownList
Introduction
AJAX (Asynchronous JavaScript and XML) has become so popular, thanks to Google Suggest. AJAX has opened the possibility to make more responsive and interactive web applications, bringing them closer to Windows form applications. Web developers are the bunch of guys who were happy at first. They have a new toy, which is composed of old toys they have neglected so far, and now they can make cool things with the toy. On the other hand, after getting their free account in GMail, end users demand more with their department web application. They hate postback, they want no refresh, and everything just stays there but still gets up-to-date. Is it possible?Those demanding users are my motivation to create this custom control. Let me introduce
AjaxDropDownList
, my first attempt to contribute into AJAX world. AjaxDropDownList
is a dropdownlist control that has the following features:- Fetch data asynchronously in the background from a server source, with no postback.
- Can trigger change event to other dropdownlists, thus generating a cascading linked dropdownlist effect.
- Is encapsulated into a single control which can be easily dragged and dropped into the designer or added from the server code.
- Uses common code to access the
selectedItem
just like a normal dropdownlist, thus can be easily integrated with other UI framework. - Compatible with Internet Explorer 6, Mozilla Firefox 1.04, and Netscape 8.02.
AjaxDropDownList
, it does not use XML to transfer the data. I use JSON (JavaScript Object Notation) by Douglas Crockford which is more lightweight and can be easily consumed by JavaScript as an object. I am aware of the potential and flexibility that XML offers compared to JSON. But in my case, JSON is enough to serve the requirements.How to use the control
Starting from the sample project
Download and open the project in VS.NET 2003. Edit GetLookupData.aspx.cs and change the connection string to point to a valid NorthWind database. If for some reason you don�t have NorthWind database, you can download the database script from Microsoft. Just do a search on Google.Once the solution is built successfully, run and browse the default.aspx. Evaluate if this is the control you are looking for.
What the demo is about
In this demo page, we have threeAjaxDropDownList
s: Customers (ddlCustomers
), Orders (ddlOrders
) and Products (ddlProducts
).Orders dropdownlist depends on Customers. It means if we make a selection in Customers dropdownlist, then the Order dropdownlist will be filtered based on our selection. Furthermore, Products is depending on Orders. So a selection on Customers will trigger a change in Orders, and subsequently trigger a change in Products. Please note that all this happens without any postbacks.
While playing around with the dropdownlist, you may encounter a JavaScript alert box showing an error message. It could mean a lot of things, but most probably your connection to the database is not working.
Now press the Submit button and the page will finally do a postback. The selected text of each dropdownlist will be shown on the right hand side. This is to demonstrate that the standard code to access the selection in ASP.NET
DropDownList
still applies to AjaxDropDownList
.What is in there
There are three important parts:- AjaxDropDownList.cs It contains the custom control. Put this file in a separate Class Library project so that we can add the control to the toolbox without any trouble.
- Default.aspx It is a sample web page that uses
AjaxDropDownList
controls. The controls are dropped into the designer from the Toolbox.
To addAjaxDropDownList
into your toolbox, right click on the toolbox pane, and then select Add/Remove Items. It will bring up Customize Toolbox dialog. Press Browse button then select CustomControl.dll or the assembly that containsAjaxDropDownList
. A big list of control with checkboxes will appear. Ensure thatAjaxDropDownList
is selected and close the dialog. The control will appear in the toolbox, ready to be dropped to the designer. - GetLookupData.aspx This is the page that handles the request from
xmlHttp
and returns the appropriate JSON. It needs to handle two query strings:- �
id
� is the lookup name or identifier, e.g., Country, Currency, Order, Product, and InvoiceStatus. - �
filter
� (optional) describes the filter in name-value pair, e.g., Customer, ALFKI.
http://localhost/GetLookupData.aspx?id=Orders&filter=Customer,ALFKI
Translates into:�Get data from Orders for Customer code = ALFKI�
It is up to you how you want to implement the request handler.
Warning: GetLookupData.aspx is just a simple example which is not suitable for a product environment. - �
Code Walkthrough
AjaxDropDownList
control uses JavaScript intensively. The JavaScript code is embedded into the control and will be injected into the response stream when the control is rendered. In order to minimize HTTP payload, the JavaScript code has been minimized using JavaScript Minifier. However, this will make debugging on the real code difficult and frustrating.Therefore, in the sample project I provide the source code in its original format and all comments still in place. Please refer to SourceScript.aspx during this walkthrough.
XMLHTTP
JavaScript code utilizesxmlHttp
object to make requests to the web server either asynchronously or synchronously. As the request can be made without refreshing the page, the web page looks more responsive and interactive. The method getXMLHTTP()
is called to a get reference to the xmlHttp
object, regardless of how the browser implements this object.Controller
EveryAjaxDropDownList
is rendered as a <SELECT>
element in HTML. Each of these elements has its controller, called AjaxDropDownController
. The controller has a lot of things to do:- Execute asynchronous request to web server to get data.
- Populate the dropdownlist.
- Listen to the change event of dropdownlist.
- Be the observer and the observable.
- Persist the content of dropdownlist in the client side.
<SELECT>
element as the view, the data that resides in the web server as the model, and the controller itself as the controller, although I don't want to emphasize this pattern as it is not fully implemented.Performing asynchronous background request
When the controller needs to update its dropdownlist, it will callload()
which in turn calls getSource()
. While calling these methods, it may pass a filter string, which is the name-value pair of the dropdownlist that it depends to. Inside the getSource()
method, a request URL is constructed which contains the id
and filter
parameters.var requestUrl = baseUrl + "?id=" + self.lookupName;
if (filter != undefined && filter != "")
{
requestUrl += "&filter=" + filter;
}
Then after a reference to the xmlHttp
object is secured, it will send the request. Note the last parameter in xmlHttp.open
which is set to true
to indicate an asynchronous request.xmlHttp = getXMLHTTP();
if (xmlHttp)
{
xmlHttp.onreadystatechange = doReadyStateChange;
xmlHttp.open("GET", requestUrl, true);
xmlHttp.send();
}
As the nature of the request is asynchronous, we could not determine when the response will be available. Therefore, we assign an event handler doReadyStateChange
to the onreadystatechange
property. This event handler will be called each time the state of the request changes.function doReadyStateChange(){
if (xmlHttp.readyState == 4)
{
if (xmlHttp.status == 200)
{
eval("var d=" + xmlHttp.responseText);
if (d != null)
{
populateList(d);
}
}
else
{
alert("There was a problem retrieving the XML data:\n" +
xmlHttp.statusText);
}
}
}
}
In doReadyStateChange
, we check for readyState = 4
which means "complete" and status= 200
which indicates an "OK" HTTP status code. Once these conditions are satisfied, it is time to process the response stream.Processing the response stream
As mentioned earlier, I used JavaScript Object Notation (JSON) to transfer data from the web server to the client. JSON is more lightweight than XML and can be easily converted into JavaScript object hierarchy.For example, when we send a request like this:
http://localhost/GetLookupData.aspx?id=Product&filter=Order,10280
The server will return:[{"value":"24","name":"Guaran� Fant�stica"},
{"value":"55","name":"P�t� chinois"},
{"value":"75","name":"Rh�nbr�u Klosterbier"}]
We concatenate the responseText
with another string to make a valid JavaScript statement and execute the statement using eval
.eval("var d=" + xmlHttp.responseText);
As a result, an in-memory object hierarchy is created as follows:d +--- [0] +--- value: 24
| |--- name: Guaran� Fant�stica
|
+--- [1] +--- value: 55
| |--- name: P�t� chinois
|
+--- [2] +--- value: 75
|--- name: Rh�nbr�u Klosterbier
We can traverse the object hierarchy with ease, for example:d[0].value
will return 24.d[2].name
will return Rh�nbr�u Klosterbier.d.length
will return 3.
populateList()
, which is responsible to populate the corresponding <SELECT>
element. The populateList
will first clear the option items from the select
, then iterate through the object hierarchy and create new option items.Observer pattern
AjaxDropDownController
implements the observer pattern. An instance of AjaxDropDownController
can be both the observer and the observable. Each instance keeps a list of observers, so that when the value changes in the corresponding dropdownlist, the controller can notify each observer about the changes.The list of observers is kept in this array:
var observers = [];
Methods
addObserver()
This method is called to add a new observer into the list. Before adding into the list, it will check whether the new observer is already in the list, thus ensuring uniqueness..removeObserver()
- omitted. A complete implementation of the observer pattern requires this method. This method is called to remove an observer from the list. Currently, I don't see any practical usage so I don't implement the method.notify()
We call this method to notify all observers about the changes in the the corresponding dropdownlist. This method will construct a filter string based on theselectedIndex
of the dropdownlist, then iterate through the observers list and call theload()
method.load()
When this method is called, the controller will callgetSource
to get the source data asynchronously usingxmlHttp
.
Persisting content
When the content of the dropdownlist is changed in the client side, the change is not carried to the server side during postback. Therefore, we could not use the standard code to access the selected item in the dropdownlist, e.g. usingSelectedItem
or SelectedIndex
of the dropdownlist. This raises an issue when we want to incorporate the control into a data binding framework or we simply could not afford to write special code to handle the control.That is the reason why I keep the content of the dropdownlist in the client side. For every
AjaxDropDownList
, there will be one hidden field associated with it. This hidden field is the container of the dropdownlist content as a value delimited string, e.g.:24|Guaran� Fant�stica|55|P�t� chinois|75|Rh�nbr�u Klosterbier
The delimiter character is configurable in the control. Therefore, if you don't like '|', you can change to another character..During postback, the value delimited string will be read and the corresponding items are created in the dropdownlist.
The server side
Discussion about the code is not complete without touching the server side because this is where the data comes from. Unlike the client side, the server side code is relatively simple. Basically, we need to handle the request sent by the
xmlHttp
and provide the data in the form of JSON.It is up to you how you want to get the data from the database. The GetLookupData.aspx is not a good example as it is prone to SQL injection attacks. In real life situations, you may need to call the data access framework instead of directly connecting to the database. You may also need to cache the data in memory to save the roundtrip to database.
What's most important for the client side is the format of the response you return to the client side. It has to be in this format:
[ {"value":"value1","name","name1"},
{"value":"value2","name","name2"},
...
{"value":"valueN","name","nameN"} ]
where value1... valueN denote item values and name1... nameN denote item texts.The custom control
AjaxDropDownList
is a custom control that inherits from System.Web.UI.WebControls.DropDownList
. It encapsulates all code to provide dynamic data population from the client side and let the dropdownlist participate in the linked dropdownlist chain.It has four additional public properties on top of the standard properties from
DropDownList
:Observers
Is anArrayList
of observer objects. When you add anotherAjaxDropDownList
into this, theAjaxDropDownList
will become dependent on the currentDropDownList
. For example:DropDownList1.Observers.Add(DropDownList2);
It will makeDropDownList2
dependent onDropDownList1
.SourceUrl
Set the URL of the source data withoutid
andfilter
parameters, e.g. http://localhost/getLookupHandler.aspx.LookupName
Identifier of the lookup list associated with the currentDropDownList
, e.g. Country, InvoiceStatus. This identifier will be passed asid
parameter in the request, for example:http://localhost/getLookupHandler.aspx?id=Country
It will also be used to composefilter
, for example:http://localhost/getLookupHandler.aspx?id=State&filter=Country,AU
Delimiter
A character to separate values persisted in the hidden field in the client side. Defaulted to '|'