accessing java objects with flex remoting
Query the database through Flex by invoking a server side Java object (the ItemDao) by Flex remoting with Blaze DS.
Displays the result, which is a java.util.List<Item> in a DataGrid and a ComboBox.
Now that we have successfully configured Spring with Flex, we would like to access server side Java objects, which we expose through Spring.
We will query Items from the database using Spring/Hibernate through the ItemDao.
This is how my Item.java looks like:
This is how my Item.java looks like:
@Entity
public class Item implements Serializable {
private static final long serialVersionUID = 340952899861L;
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Long id;
@Column(length = 20, unique = true)
private String code;
@Column(length = 15, unique = true)
private String name;
@Column(length = 50)
private String description;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public String getCode() {
return code;
}
public void setCode(String code) {
this.code = code;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
}
And this is how my ItemDao.java looks like:
public class ItemDao extends HibernateDaoSupport {
@SuppressWarnings("unchecked")
public List<Item> getAll() {
return super.getHibernateTemplate().loadAll(Item.class);
}
}
As is obvious, I configure the ItemDao through Spring and then expose it as below:
...
<flex:remoting-destination ref="itemDao"/>
...
The one thing to remember here is that, Flex handles remote access through call back mechanism. That is, the call to the Java object, is asynchronous. The call returns immediately. However, the result (or the error) is got asynchronously by registering suitable listeners.
The following code snippet illustrates this:
<?xml version="1.0" encoding="utf-8"?>
<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"
initialize="initData();">
<mx:Script>
<![CDATA[
import mx.controls.Alert;
import mx.rpc.remoting.RemoteObject;
import mx.rpc.events.ResultEvent;
import mx.rpc.events.FaultEvent;
import mx.collections.ArrayCollection;
[Bindable]
private var itemsMultiArray:ArrayCollection;
private function initData():void {
//call the getAll() method on the itemDao remote object, which has been exposed
var itemRO:RemoteObject = new RemoteObject();
itemRO.destination = "itemDao";
itemRO.addEventListener(ResultEvent.RESULT, getItemListResultHandler);
itemRO.addEventListener(FaultEvent.FAULT, faultHandler);
itemRO.getOperation("getAll").send();
}
private function getItemListResultHandler(event:ResultEvent):void {
itemsMultiArray = (ArrayCollection)(event.result);
}
private function faultHandler (event:FaultEvent):void {
// Deal with event.fault.faultString, etc.
Alert.show(event.fault.faultString, 'Error');
}
]]>
</mx:Script>
<mx:DataGrid id="itemDetails" width="511" dataProvider="{itemsMultiArray}">
<mx:columns>
<mx:DataGridColumn headerText="ID" dataField="id"/>
<mx:DataGridColumn headerText="Name" dataField="name"/>
<mx:DataGridColumn headerText="Code" dataField="code"/>
</mx:columns>
</mx:DataGrid>
</mx:Application>
This is how it looks like:
The most important thing to note here is that the result that is got back in the call back function, should be cast with the proper data type in Flex. This is the link which has the details under the head Converting data from Java to ActionScript. From here we get to know the correct casting. In this case, we are casting the result into mx.collections.ArrayCollection since it is of java.util.List type.
Also note that though each element in the array is of the type Item, it can be accessed by its bean attribute name. That is how we can access the id, name and code in the DataGrid. Note that you cannot invoke the getter method like getId() on the object, it will not work.
So far, we have used a DataGrid which can extract data based on the attribute names. What if we use a ComboBox which needs a two dimensional array with data and label attributes? This is the code snippet of the modified getItemListResultHandler():
...
[Bindable]
private var itemComboArray:Array = new Array();
...
private function getItemListResultHandler(event:ResultEvent):void {
itemsMultiArray = (ArrayCollection)(event.result);
var count:int = 0;
for each (var item:Object in itemsMultiArray) {
var content:Object = new Object();
content["label"] = item.name;
content["data"] = item.id;
itemComboArray[count++] = content;
}
}
And this is the code to add the ComboBox:
...
<mx:ComboBox id="itemCombo" dataProvider="{itemComboArray}"></mx:ComboBox>
...
The screen looks like this now: