AJAX Made Simple With DWR
AJAX Made Simple With DWR
AJAX Made Simple With DWR
Sample application
The example application used in this article simulates an apartment
rental search engine for the city of Toronto. The user can select a set
of search criteria before performing the search. To improve interaction,
AJAX is used in two occasions:
• The application notifies the user of the number of search results that
match his selection. This number is updated—using AJAX—as the user
selects the amount of desired bedrooms and bathrooms, and the price
range. By showing the user the number of results, the user won't need
to hit the search button when no results or too many results match the
user's criteria.
• The database query that retrieves the units from the database is
performed using AJAX. The database search executes when the user
presses the Show Results button. Thus, the application seems more
responsive, as the whole page doesn't need to be reloaded to show the
results.
Database
The database we use is HSQL, a Java SQL database engine with a
small footprint, which can be bundled with the Web application with no
additional installation and configuration. A SQL file is used to create
the in-memory table and add some records when the Web application
context is started.
Java classes
The application contains two main classes called Apartment and
ApartmentDAO. The Apartment.java class is a simple Java class with
attributes and getter/setter methods. ApartmentDAO.java is the data-
access class used to query the database and get information based on
the user's search criteria. The implementation of the ApartmentDAO
class is straightforward; it uses straight Java Database Connectivity
calls to get the total number of units and the list of available units
matching the user's request.
DWR configuration and use
Setting up DWR for use is easy: Copy the DWR jar file to the Web
application's WEB-INF/lib directory, add a servlet declaration to
web.xml, and create the DWR configuration file. The DWR distribution
only requires the use of a single jar file. You must add the DWR servlet
into the application's deployment descriptor in WEB-INF/web.xml:
<servlet>
<servlet-name>dwr-invoker</servlet-name>
<display-name>DWR Servlet</display-name>
<description>Direct Web Remoter
Servlet</description>
<servlet-
class>uk.ltd.getahead.dwr.DWRServlet</servlet-class>
<init-param>
<param-name>debug</param-name>
<param-value>true</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>dwr-invoker</servlet-name>
<url-pattern>/dwr/*</url-pattern>
</servlet-mapping>
Now you must let DWR know what objects will receive requests
through the XMLHttpRequest object. Accomplish that task by using
DWR's own configuration file called dwr.xml. In the configuration file,
you define the objects that DWR will allow you to call from your HTML
pages. By design, DWR allows access to all the exposed class's public
methods, but in our example, we only allow access to a few methods.
Here's the configuration file for our example:
<dwr>
<allow>
<convert converter="bean"
match="dwr.sample.Apartment"/>
<create creator="new" javascript="ApartmentDAO"
class="dwr.sample.ApartmentDAO">
<include method="findApartments"/>
<include method="countApartments"/>
</create>
</allow>
</dwr>
The above file accomplishes two goals in our example. First, the
<convert> tag tells DWR to convert objects of the
dwr.sample.Apartment type into JavaScript associative arrays,
because, for security reasons, DWR doesn't convert regular beans by
default. Second, the <create> tag makes DWR expose the
dwr.sample.ApartmentDAO class to be called from JavaScript; the
JavaScript file we use in our pages is defined by the javascript
attribute. We must pay attention to the <include> tags, as those
identify which methods from the dwr.sample.ApartmentDAO class will
be made available.
HTML/JSP code
Once the configuration is done, you can start your Web application,
and DWR will be ready to call the methods you need from your
HTML/JavaServer Pages (JSP) page, without you creating JavaScript
files. In the search.jsp file, we must add references to the JavaScript
interface provided by DWR, as well as the DWR engine, by adding
three lines to our code:
<script src='dwr/interface/ApartmentDAO.js'></script>
<script src='dwr/engine.js'></script>
<script src='dwr/util.js'></script>
The first use of AJAX in the example application can be noticed when
the user changes the search criteria; as he can see, as the criteria
changes, the number of available apartments is updated. I created two
JavaScript functions: The updateTotal() function is called when a
value in one of the select boxes changes. The
ApartmentDAO.countApartments() function is the most important
piece. Most interesting is the first parameter, the loadTotal()
function, which identifies the callback method that DWR will use when
it receives a response from the server. loadTotal() is then called to
display the result on the HTML page's <div>. Here's how the
JavaScript functions are used in the described interaction scenario:
function updateTotal() {
$("resultTable").style.display = 'none';
var bedrooms =
document.getElementById("bedrooms").value;
var bathrooms =
document.getElementById("bathrooms").value;
var price = document.getElementById("price").value;
ApartmentDAO.countApartments(loadTotal, bedrooms,
bathrooms, price);
}
function loadTotal(data) {
document.getElementById("totalRecords").innerHTML =
data;
}
Obviously, the user will want to see the list of apartments that satisfy
his search. So, when the user is satisfied with the criteria and total
apartments available, he presses the Show Results button, which calls
the updateResults() JavaScript method:
function updateResults() {
DWRUtil.removeAllRows("apartmentsbody");
var bedrooms =
document.getElementById("bedrooms").value;
var bathrooms =
document.getElementById("bathrooms").value;
var price = document.getElementById("price").value;
ApartmentDAO.findApartments(fillTable, bedrooms,
bathrooms, price);
$("resultTable").style.display = '';
}
function fillTable(apartment) {
DWRUtil.addRows("apartmentsbody", apartment, [ getId,
getAddress, getBedrooms, getBathrooms, getPrice ]);
}
The updateResults() method clears the table area assigned to hold
the search results, gets all the necessary parameters from the UI, and
passes those parameters to the ApartmentDAO JavaScript object
created by DWR. The database query is then performed, and the
callback function fillTable() is called, which then parses the object
returned by DWR, and prints them to the page.
Security concern
To keep the example brief, the ApartmentDAO class was kept as simple
as possible, but such a class will usually have a set of methods to
manipulate data, such as insert(), update(), and delete(). DWR
exposes all public methods to any calling HTML page. For security
reasons, it's not wise to expose your data-access layer methods in
such a way. A developer could create a facade to centralize the
communication between the calling JavaScript function and lower-level
business components, thus limiting the exposed functionality.
Conclusion
This article's example only gets you started using AJAX with DWR in
your own projects. DWR helps you keep focused on how to improve
your application's interaction model, removing the burden that comes
with coding and debugging JavaScript code. The most interesting
challenge of using AJAX is defining where and how to improve
usability. DWR helps you focus totally on how to make your application
more user-friendly by handling the communication between the
Webpage and your Java objects.