As we were designing Norfolk AIR it was apparent that the standard single mapping interface was going to be a thing of the past. If one is cool 2 must be better!
We were really worried about the amount of real estate 2 map windows would consume but I think the interface guys did a brilliant job on the layout and our users are raving about it.
So know that we have two it opens up a whole lot of functionality that would've been hard to accomplish in a single map frame design.
I give you the imagery interface window..........................
The interface consist of an ESRI .NET arcserver mapping control, an ASP.NET panel control and a standard ASP.NET dropdownlist.
First order of business was to keep both the mapping interface (discussed in the next post) and the imagery interface in sync. Things like pan one and the other follows. Zoom one and the other follows. etc.
This was accomplished with a little code behind in the Map_ExtentChanged on the server.
Protected Sub Map1_ExtentChanged(ByVal sender As Object, ByVal args As ESRI.ArcGIS.ADF.Web.UI.WebControls.ExtentEventArgs) Handles Map1.ExtentChanged
Dim env As ESRI.ArcGIS.ADF.Web.Geometry.Envelope = args.NewExtent
Dim env2 As ESRI.ArcGIS.ADF.Web.Geometry.Envelope = Map2.Extent
If env2 IsNot Nothing Then
If Not Map2.Extent.IsEqual(env) Then
Map2.Extent = env
Map1.CallbackResults.CopyFrom(Map2.CallbackResults)
End If
End If
End Sub
The above code will basically eval the current extent of the control and test to see if it is equal to the other map control. Although this isn't dynamic (ie both maps move at the same time) it is pretty quick and the users haven't complained. In the future I may look at seeing if this can be done dynamically. So for know a callback is required.
So what does this window do?
It allows users to toggle 4 years of imagery(we are adding more), display the location in nice google streetview (nice) and fire of a sub window that will tile all years of imagery together.
As the dropdown is changed, imagery requests are sent as an AJAX call to the server that updates the MapResource for the ArcGIS server Map control. When this was first developed it was a postback(slow, refresh, click, yuk). Introducing AJAX allowed us to just target this specific map control without any refresh. We used the standard callback framework for the map controls that ESRI provided and built a PageLoadingHandler. This was a bit tricky as sometimes the callback wouldn't result in a map change(ie show the google window) so we are building the callback and then evaling it on the client. If it is just a standard map callback then javascript just runs it but if it is something else (ie google, errors, open the all image subwindow) then javascript is used to handle those request.
function PageLoadingHandler(sender, args) {
var dataItems = args.get_dataItems();
if (dataItems['Map2'] != null)
//alert(dataItems['Map2']);
//alert(dataItems['DropDownList1']);
{
if (dataItems['Map2'] == 'error')
{
errorHandler();
}
else if (dataItems['Map2'] == 'showstreetview')
{
loadStreetView();
showStreetView();
}
else if (dataItems['Map2'] == 'allimage')
{
var w = dataItems['DropDownList1'];
// alert(w);
showAllImage(w);
}
else
{
hideStreetView();
processCallbackResult(dataItems['Map2'], 'Map2');
}
}
}
So as you can see from the above javascript, that we handle pretty everything in the AJAX call with javascript. We can call our subfunctions, like if the user has selected 'Block View' the server will pass a statement 'showstreetview'. This will be caught by the PageLoadingHandler and will call the loadstreetview() and showstreetview() functions.
In this case it is a nice overlayed div with the current address in google streetview
We have found using the geocoder from google that sometimes are house is down the street thus the term "Block View". We are researching using an address location and converting the geometry to pass the exact location to google but for know it is "Block View".
For the future we are looking at integrating Pictometry images into this window as well being able to see a 360 (north, south, east, west) oblique image of the property.
At first I was a bit skeptical about the 2 window design but I must say it has turned out to be one of the most productive designs that I have run across.
-chris
Wednesday, March 18, 2009
Tuesday, March 3, 2009
Front End Search for Norfolk AIR
So the most often asked question I have received so far is how did we handle the search function that is embedded into Norfolk AIR. Being in the GIS development biz it has been something that was always on the radar but just never had the right technology to pull of. Ah and in strolled Ajax
This little beauty is probably the most significant part of the website. Simple, straightforward, so easy anyone can use it. Just start typing and we will try and find it for you.
How was it built?
The search box is actually just a simple asp.net textbox with an ajax autocomplete extender, a textboxwatermark extender and and a hiddenField control all wrapped into one big user control called SearchAssist.
Tied to the search AutoCompleteExtender is a web service that we use to filter the results to the text box (as well as a service other application searchs).
So it goes like this. As the user types the autocomplete extender is listening for the 3rd keystroke. Once the 3rd keystroke is complete the system accesses the webservice to try and retrieve the qualifying values. If it finds some it will return a maximum of 10 records that the user can select from. Pretty simple!
Yes and No
The pretty yes. The simple no.
Behind the scenes we have 3 different sql databases that this search pulls from. (master address, gis (gpins) and real estate (account number)). Actually we are getting ready to add a 4th common place name. Anyway, so as the user is searching the web services is trying to locate like values from 3 different database and present them back to the user...err.. quickly. The quickly part is the key. How do you quickly query 3 database (minimum of 100,000 records each) as fast as the user can type? yeh. The answer is you don't. To overcome the latency problem we actually front ended the queries with a bunch of logic to try and determine if we can narrow down the search before querying the server. Things like if the 1st character isn't a number then I know I don't have to search the address database. If I know I have 5 numbers in a row I don't look for a house number it is probably a gpin or account.
Although a big pain to come up with this was a lifesaver in terms of speed. As the project has progressed I have been able to refine this logic more and more so the query is usually just to one database. Oh yeah did I mention the indexing?
The next big hurdle was know that the user has found a value that they were looking for how does the system know which database that it came from. Don't want to perform all those searches again do I?
Ajax's autocomplete extender actually has a nice feature that I didn't find a lot of documentation on but was able to implement in that you can not only set the returned item name but you can also tie a value to it. This worked out nice as we were able to not only return the search values but also the type(which database they came from)
items.Add(AjaxControlToolkit.AutoCompleteExtender.CreateAutoCompleteItem(dt.Rows.Item(i).Item(displayField).ToString, searchType))
Once the user selects an item from the list we run a little bit of javascript code embedded in the user control(OnClientItemSelected) that will place the database type to a hidden value (thats what the hidden field on the control is for) on the form and presto. We know have the value the user was looking for as well as what database it came from all accessible on the server.
This little beauty is probably the most significant part of the website. Simple, straightforward, so easy anyone can use it. Just start typing and we will try and find it for you.
How was it built?
The search box is actually just a simple asp.net textbox with an ajax autocomplete extender, a textboxwatermark extender and and a hiddenField control all wrapped into one big user control called SearchAssist.
Tied to the search AutoCompleteExtender is a web service that we use to filter the results to the text box (as well as a service other application searchs).
So it goes like this. As the user types the autocomplete extender is listening for the 3rd keystroke. Once the 3rd keystroke is complete the system accesses the webservice to try and retrieve the qualifying values. If it finds some it will return a maximum of 10 records that the user can select from. Pretty simple!
Yes and No
The pretty yes. The simple no.
Behind the scenes we have 3 different sql databases that this search pulls from. (master address, gis (gpins) and real estate (account number)). Actually we are getting ready to add a 4th common place name. Anyway, so as the user is searching the web services is trying to locate like values from 3 different database and present them back to the user...err.. quickly. The quickly part is the key. How do you quickly query 3 database (minimum of 100,000 records each) as fast as the user can type? yeh. The answer is you don't. To overcome the latency problem we actually front ended the queries with a bunch of logic to try and determine if we can narrow down the search before querying the server. Things like if the 1st character isn't a number then I know I don't have to search the address database. If I know I have 5 numbers in a row I don't look for a house number it is probably a gpin or account.
Although a big pain to come up with this was a lifesaver in terms of speed. As the project has progressed I have been able to refine this logic more and more so the query is usually just to one database. Oh yeah did I mention the indexing?
The next big hurdle was know that the user has found a value that they were looking for how does the system know which database that it came from. Don't want to perform all those searches again do I?
Ajax's autocomplete extender actually has a nice feature that I didn't find a lot of documentation on but was able to implement in that you can not only set the returned item name but you can also tie a value to it. This worked out nice as we were able to not only return the search values but also the type(which database they came from)
items.Add(AjaxControlToolkit.AutoCompleteExtender.CreateAutoCompleteItem(dt.Rows.Item(i).Item(displayField).ToString, searchType))
Once the user selects an item from the list we run a little bit of javascript code embedded in the user control(OnClientItemSelected) that will place the database type to a hidden value (thats what the hidden field on the control is for) on the form and presto. We know have the value the user was looking for as well as what database it came from all accessible on the server.
//function SearchSelect( source, eventArgs )
//{
//alert( " Key : "+ eventArgs.get_text() +" Value : "+eventArgs.get_value());
//typeSearch = $get("<%=typeSearch.ClientID %>");
//typeSearch.value = eventArgs.get_value();
//}
That is it. How we were able to produce a really quick, easy to user interface that allows you to search our entire system. The search is so fast that I haven't had anyone be able to out type it yet.
cj
Subscribe to:
Posts (Atom)