Friday, December 11, 2009

ESRI Flex Map with Robotlegs

Well, I'm not really one to leave well enough alone and I just had to give another Flex framework a try for my ESRI Flex mapping needs. My latest victim was Robotlegs. If you look at my earlier blog postings, you might have seen me mention that my first Flex framework was Cairngorm. I didn't really dislike Cairngorm, but after some conferences and reading around, I felt like I should expand my Framework horizons. Most recently I have been using Swiz with a lot of success. I think I use Swiz in the most minimal way possible, which is actually something for a future blog posting.

But I had also tried at one point to give Robotlegs a try. I had worked through some of the docs, but had not really dived into any of the examples. So I finally decided that I needed to get my feet wet a bit and dived into some reading material. Joel Hooks had a posting of a presentation that I particularly enjoyed. So after some trial and error, I think I wrapped my head around the basics of Robotlegs. Like Swiz, Robotlegs utilizes metadata to support Dependency Injection in the framework. I'll be honest here, I'm spoiled, I don't know how to do dependency injection in Flex/AS3 manually. Adding that to my list of things to learn. Anyway, this makes tying classes and events together pretty simple.

After a couple of hours of watching videos and creeping through some examples, I was able to come up with this example, with view source enabled of course. Now, as far as Flex Mapping applications go, this is about as simple as it gets and as far as I wanted to go tonight as it's getting late. But the basics are in there. You have a map, with it's layers and extent tied to a model. You have a datagrid tied to a custom ListCollection ( I took portions of this from another project I'm working on ) and a graphicslayer that gets highlighted when you click on the datagrid. And of course a QueryTask to handle getting attribute information. Most folks that work with Flex Maps using the ESRI Flex API will be familiar with this and realize this is probably the starting point for about 90% of your projects.

Now on to my impressions of Robotlegs. I will say this, Robotlegs got me in the mindset of the Single Responsibility Principle when I was going through my project. For example, I have a command to query the States Service and a command o handle the results. In most of my recent work, I had one controller doing both those tasks. Now, I'm not going to go "programmer purgatory" for using my controllers that way and the only coding cookie I'll get for wanting to follow that principle is the sugar cookies my wife and kids make, but it is one of those things I have found makes maintaining and testing of code easier. My only complaint would be is that I found, even looking at examples, is that the code started looking a little "Cairngormish". By that I mean, I had to manually wire my events to my commands. This started looking an awful lot like the FrontControllers I'd seen in previous apps.

One other thing bothered me, but I'm sure it has more to do with my following examples as opposed to really knowing how to use the framework is the need to override functions and extend Robotlegs classes for everything. Something I have been trying to do, is keep 3rd party APIs that aren't specific to the output of my app as minimal as possible. That's actually a topic for a later post, but it was something I found odd as I worked through my first real attempt. But like I said, I have a feeling that as I get more familiar with Robotlegs, I can find a nice balance to keep the framework implementation as minimal as possible. Something I really liked was the Mediator for views. I have been using the Presentation Model on my most recent work app and I'm a big fan of using a model for my view that can be tested independently. The Mediator class in Robotlegs is a very nice way of keeping your views as "dumb" as possible.

So, that's it for my first run at Robotlegs. I'm going to dive into it some more and try some things to see how I like using it most. I'm sure the more I use it, the more I'll enjoy it. It's nice having another tool in my Flex toolbox that I can pound on something with.

Wednesday, December 9, 2009

Highlight Map from List Items

I was inspired by this posting on the ESRI ArcGIS Server Blog filed in the Flex tag, that linked your graphics on a map with a DataGrid. This is a useful function and one I thought could be encapsulated into it's own task. In my case, I just had a need to highlight a graphic on my map that was associated with a list. I did not need to highlight the list when I hovered over the map because I have designed my Graphics with tooltips and felt that two-way binding would be "too much" flash for business purposes. You certainly encapsulate that function and if I were to do so, would probably build it into a utility type of class. With that said, here is the code.
package net.odoe.model.graphic
{
import com.esri.ags.Graphic;
import com.esri.ags.geometry.Geometry;
import com.esri.ags.layers.GraphicsLayer;
import com.esri.ags.symbol.SimpleFillSymbol;
import com.esri.ags.symbol.SimpleLineSymbol;
import com.esri.ags.symbol.SimpleMarkerSymbol;
import com.esri.ags.symbol.Symbol;

public class HighlightGraphic extends Graphic
{
protected var _defaultSymbol : Symbol = new Symbol();

/**
* This function will save the default graphic symbology
* highlight the graphic for the currently selected item.
* @param data that holds attribute information; i.e. from a ListEvent (event.itemRenderer.data).
* @param graphicsLayer that holds the graphics that can be selected.
* @return a higlighted graphic.
*
*/
public function setHighlightGraphic( data : Object, graphicsLayer : GraphicsLayer ) : Graphic {
if( graphicsLayer ) {
_defaultSymbol = findGraphicByAttribute( data , graphicsLayer).symbol;
return featureHighlight( data, graphicsLayer );
}
return new Graphic();

}

/**
* This function will reset the previously selected graphic
* to its default symbology.
* @param data that holds attribute information; i.e. from a ListEvent (event.itemRenderer.data).
* @param graphicsLayer that holds the graphics that can be selected.
* @return default graphic.
*
*/
public function resetDefaultGraphic( data : Object, graphicsLayer : GraphicsLayer ) : Graphic {
if( graphicsLayer ) {
var _graphic : Graphic = findGraphicByAttribute( data, graphicsLayer );
_graphic.symbol = _defaultSymbol;
return _graphic;
}

return new Graphic();

}

//------------------------------------------------------------------------------------------------------
//----PROTECTED METHODS---------------------------------------------------------------------------------
//------------------------------------------------------------------------------------------------------
protected function findGraphicByAttribute( attributes : Object, graphicsLayer : GraphicsLayer ) : Graphic {
for each( var graphic : Graphic in graphicsLayer.graphicProvider ) {
if ( graphic.attributes == attributes )
{
return graphic;
}
}
return new Graphic();
}

protected function featureHighlight( data : Object, graphicsLayer : GraphicsLayer ) : Graphic {
if( graphicsLayer ) {
var _graphic : Graphic = findGraphicByAttribute( data, graphicsLayer );
_graphic.symbol = highLightSymbol( _graphic.geometry );
return _graphic;
}

return new Graphic();
}

protected function highLightSymbol( geometry : Geometry ) : Symbol {
if( geometry.type == Geometry.POLYLINE ) {
var _highlightLineSymbol : SimpleLineSymbol = new SimpleLineSymbol;
_highlightLineSymbol.color = 0xFF00FF;
_highlightLineSymbol.width = 10;

return _highlightLineSymbol;
}
if( geometry.type == Geometry.MAPPOINT ) {
var _outline : SimpleLineSymbol = new SimpleLineSymbol;
_outline.style = "solid";
_outline.color = 0x0000CD;
_outline.width = 2;

var _highlightPtSymbol : SimpleMarkerSymbol = new SimpleMarkerSymbol;
_highlightPtSymbol.style = "circle";
_highlightPtSymbol.color = 0xFF00FF;
_highlightPtSymbol.size = 15;
_highlightPtSymbol.alpha = 1;
_highlightPtSymbol.outline = _outline;

return _highlightPtSymbol;
}
if( geometry.type == Geometry.POLYGON ) {
var _outlinePoly : SimpleLineSymbol = new SimpleLineSymbol();
_outlinePoly.style = "solid";
_outlinePoly.color = 0x0000CD;
_outlinePoly.width = 2;

var _highlightPolySymbol : SimpleFillSymbol = new SimpleFillSymbol;
_highlightPolySymbol.style = "solid";
_highlightPolySymbol.color = 0x00FFFF;
_highlightPolySymbol.alpha = 0.3;
_highlightPolySymbol.outline = _outlinePoly;

return _highlightPolySymbol;
}

return new Symbol();
}

}
}


I have made the functions in here protected, as I found a need at a later time to extend this class and override the protected function findGraphicByAttribute to recognize a particular attribute field (i.e. graphic.attributes.Name == attributes.Name) and not just all attributes.

I made an example of the function just like the one on the ESRI ArcGIS Server Blog using my class.
It can be found here with View Source enabled. Hope this benefits someone.