The recently released JavaFX platform [10] allows developers to build rich internet applications (RIA) that can include audio and video. Using JavaFX, it is possible to create highly interactive applications. Moreover, it is possible to easily create content for different devices (desktop, mobile phone, television, and so on). JavaFX is a compiled language, like Java, and is highly portable and is based on the familiar paradigm “Write Once, Run Everywhere.”
JavaFX is focused on the client side, and aims to improve the look and feel of Java GUIs so that users can experience more attractive interfaces. Of course, many client applications need to exchange information with a remote server. Nowadays, the HTTP protocol and XML are widely accepted as the best choices to exchange information, so we want to show how easy is in JavaFX to handle HTTP communication details and how we can parse and extract information from an XML data structure.
In the article we will assume you are already familiar with the basic notions of the JavaFX language.
While it is a compiled language, JavaFX mixes the features of scripting languages with those inherited from Java. Scripting languages allow for fast and easy application development, while JavaFX's Java-based heritage allow it to be a robust language.
JavaFX proposes a new coding paradigm: as a declarative language, it compels us to describe how we want our application to behave without describing the specific control flow, as we do with imperative languages. This paradigm is really powerful when we need to develop GUIs. The basic idea that stands behind that JavaFX GUI development model is that you "describe" what your interface should look like. There is a strict relationship between the code and the "visual structure." Moreover, the order used to declare objects, in the code, reflects the order used to display them. The overall result is an elegant way to create a GUI in fewer lines of code; this makes applications easier to understand and maintain.
Another interesting feature of JavaFX is that it is a statically typed language, meaning the data type of every variable, function, and so on is known at compile-time. See the Resources section [9] for links to JavaFX tutorials that explore this trait further.
To develop an application using HTTP protocol and XML, JavaFX provides several packages, which are shown below:
javafx.io.http for handling HTTP
communicationjavafx.data.pull and javafx.data.xml
for XML parsingThe class diagram in Figure 1 shows the classes contained in these packages.
[11]
Figure 1. Defining the channel rule for the RDBMS Event Generator (click for larger
view)
To handle the HTTP protocol, we can use HttpRequest
class in the javafx.io.http package. This class makes
asynchronous HTTP requests to a remote server that supports the HTTP
protocol. The HTTP methods currently supported are:
GETPOSTPUTDELETEThis class is neutral in respect to the data format exchanged, so
we can invoke a remote server and send whatever type of
information we like, as long as we supply an
OutputStream containing the data that must be sent,
using the POST or PUT HTTP methods.
The HttpRequest operation, related to each HTTP
method supported, has a specific lifecycle. We focus our attention
on the lifecycle in the case of HTTP GET method; for other methods
(POST, PUT, DELETE), the lifecycle is very similar. In the
case of an HTTP GET request, the lifecycle is shown in Figure
2.
[12]
Figure 2: HTTP GET method request lifecycle (click for larger
view)
As we can see from the diagram above, each state of the
lifecycle is defined by a specific value of the internal variables
of the HttpRequest class. Related to each variable
transition, there is a corresponding method that is called during
the transition itself, so that we can control and handle different
states in the HTTP lifecycle. These methods have the same name of
the corresponding variable, prepended with on. For
example, if we want to track when the request is trying to connect
to the server, we will use the onConnecting
function.
It is time we start coding our JavaFX HTTP client. First of all we have to declare a variable that contains our URL:
def url : String = "http://www.java.net";
Then we create the HTTP request and define our callback function, which is called when the HTTP request starts connecting.
HttpRequest {
location: url;
onConnecting: function() {
java.lang.System.out.println("Connecting");
}
}.enqueue();
Notice the method enqueue() that makes the
request.
Now we want to read the response body. We can do that using the
InputStream provided by the function
onInput. We need to add this piece of code to our
client.
onInput: function(is: InputStream) {
try {
var responseSize : Integer = is.available();
java.lang.System.out.println("Response size {responseSize}");
}
finally {
is.close();
}
}
The last step is to handle any exceptions that can occur during
the HTTP request. The HTTPRequest has a function that
is called whenever an exception occurs. So we can add the
exception-handling code below to our client.
onException: function(ex : Exception) {
System.out.println("Error: {ex.getMessage()}");
}
If you run the client using NetBeans, you should see output similar to Figure 3:

Figure 3: Client log
In the package javafx.io.http, there are two other
classes called HttpHeaders and
HttpStatus. The first class defines a set of constants
that map the corresponding HTTP header value names. The second
class defines a set of constants corresponding to the possible HTTP
response codes.
As we said, many clients today send data over HTTP using an XML format, and JavaFX offers the capability to easily parse an XML document. We focus our attention now on the other two packages, shown before in Figure 1:
javafx.data.xmljavafx.data.pullThe package javafx.data.pull contains the classes
to parse an XML document, while the javafx.data.xml
package defines some constants and handles qualified names. The
parser is event-based (similar to the
SAX parser [13]) and it supports two different data formats:
For this article, we'll focus our attention on the XML data format.
The PullParser class, the heart of JavaFX's
document parser, accepts several attributes that can be used to
control the parser. First of all, we need to declare the document
type we want to parse, which we do by using the class attribute
documentType. This string can have two values:
PullParser.XML is used for parsing XMLPullParser.JSON is used for parsing JSONAfter we declare the document type, we need to supply the input
document to parse. The parser accepts an input stream, and as we
will see later, this is very handy when we need to parse an XML
document retrieved from an HTTP request. To declare the input
stream we need to set the value of the input
variable.
So it is time we create an instance of our
PullParser, as shown below:
parser = PullParser {
documentType: PullParser.XML;
input: xmlFileInputStream;
}
While the parser analyzes the document, it generates a set of
events. We need to implement a callback function to be called in
response to these events. The callback function is called
onEvent and in its body, we implement our logic to
extract information from the document, which we will do later.
The function signature is onEvent(event : Event),
where the Event class belongs to the package
javafx.data.pull. This class contains all the
information related to the pull-parsing event, and we can use it to
extract the information we need. The type declares the
type of event, as one of the values defined in
PullParser. We are interested in the following types of
events:
START_DOCUMENT: This event is generated at the
beginning of document parsing.START_ELEMENT: This event is generated when the
parser finds a new starting element. We can use this event to read
the element attribute.END_ELEMENT: This event is generated when the
parser finds the end of the element. We can use it to read the text
contained in the element.END_DOCUMENT: This event is generated when the
parser reaches the end of the document.There are other events that can be used for JSON
documents; if you're interested, have a look at the
PullParser [14] documentation. At any rate, here's an
onEvent skeleton implementation to react to the
START_ELEMENT and END_ELEMENT events.
onEvent: function(event : Event) {
/* We start analyzing the different event types */
if (event.type == PullParser.START_ELEMENT) {
/* Here we implement our logic to handle the start element event,
for example to extract the attribute values and so on */
}
else if (event.type == PullParser.END_ELEMENT) {
/* Here we implement our logic to handle the end element */
}
}
During the parsing process, some errors can occur. We can manage
them verifying the type of Event generated by the
parser.
Now that we have described these two APIs, it is time we look at the most interesting part: how we can integrate everything so that we can code a complete XML-over-HTTP client. This can be useful if we want to have a client that exchanges information with a remote server.
Let's suppose that our JavaFX client application invokes a servlet that returns an XML file with the structure shown below:
<?xml version="1.0" encoding="UTF-8"?>
<data>
<person id="1">
<name>Mikey</name>
<surname>Mouse</surname>
</person>
</data>
This is a simple XML file, but it is enough for the purpose of
our example. Our goal is for our client to connect to the test
servlet and retrieve the XML content, and then parse it and show the
extracted information. To do that, we need to change the
HttpRequest function onInput so that
when we start receiving the XML document we parse it, too. The code
below shows how to do it:
onInput: function(is: InputStream) {
try {
PullParser {
input: is;
onEvent: function (event : Event) {
// We handle the event
}
}.parse();
}
finally {
is.close();
}
}
Notice how we have added the PullParser to the
onInput function, and that we set the parser input
stream to the one received from the HttpRequest. Now
we just need to handle the events as we described before:
....
if (event.type == PullParser.START_ELEMENT and event.level == 1) {
java.lang.System.out.println("Start a new element {event.qname.name}");
var qAttr : QName = QName {name : "id"};
var attVal : String = event.getAttributeValue(qAttr);
java.lang.System.out.println("Attribute ID value {attVal}");
}
else if (event.type == PullParser.END_ELEMENT) {
var nodeName : String = event.qname.name;
java.lang.System.out.println("End element {nodeName}");
// Now we extract the text only if the node is name or surname
if (nodeName == "name" or nodeName == "surname") {
var textVal : String = event.text;
java.lang.System.out.println("Text {textVal}");
}
}
....
It is useful to analyze the code step by step. In the case of a
PullParser.START_ELEMENT event, we use the
event.level variable. This tells us at which line the
event occurs (starting from zero, the XML document root). We know
already that the id attribute is present only on the
first line, so we limit the extraction to this line only. Then we
create a QName object setting, the name
variable to our attribute name, and then we extract the value.
In the case of PullParser.END_ELEMENT, we want to
extract the node content. To do this, we use the text
variable that contains the node value.
If everything works properly we will see the parsed items in the console, as shown in Figure 4.

Figure 4. HTTP request with XML parsing
In this article, we explored some essential features of JavaFX, focusing our attention on two important aspects: XML and HTTP. We discovered how easy is to develop a simple client that makes an HTTP request and parses the XML response. This is a basic example, but it can be further expanded adding other features; for example, connecting to a site and retrieving pictures.
Links:
[1] http://www.java.net/author/francesco-azzola
[2] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing
[3] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#javafx-basic-language-concepts
[4] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#javafx-http-and-xml-package-overview
[5] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#http-and-javafx
[6] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#xml-api
[7] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#integrating-the-http-and-xml-apis
[8] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#conclusion
[9] http://www.java.net/article/2009/02/12/javafx-http-networking-and-xml-parsing#resources
[10] http://www.javafx.com
[11] http://www.java.net/images/2009/02/pack.png
[12] http://www.java.net/images/2009/02/http_life_cycle.png
[13] http://java.sun.com/javase/6/docs/api/javax/xml/parsers/SAXParser.html
[14] http://java.sun.com/javafx/1/docs/api/javafx.data.pull/javafx.data.pull.PullParser.html
[15] http://www.java.net/today/2009/02/17/http-client-fx-source.zip
[16] http://www.sun.com/software/javafx/
[17] http://java.sun.com/javafx/1/docs/api/
[18] http://javafx.com/
[19] http://java.sun.com/javafx/1/tutorials/ui/syntax/index.html
[20] http://java.sun.com/javafx/1/tutorials/core/
[21] http://weblogs.java.net/blog/joshy/archive/2007/09/javafx_javafx_s.html