Proposal #5: Geocoder based on WFS spatial selectors - geosolutions-it/MapStore GitHub Wiki
Overview
New widget based on geocoder spatial selectors to perform simple location searches by WFS
Proposed By
Alejandro Díaz (alediator)
Assigned to Release
MapStore 1.6
State
Choose one of: Under Discussion, In Progress, Completed, Rejected, Deferred
Motivation
Let us show a simple Geocoder based on two related WFS layers:
- one for the street name
- one for the street number
Proposal
The first aproach is available here.
You can configure it adding this plugin with this configuration:
{
"ptype": "gxp_spatial_selector_geocoder",
"outputTarget": "west",
"text": "Geocoder",
"layoutConfig":{
"xtype": "form",
"buttonAlign": "right",
"autoScroll":true,
"frame":true
},
"crossParameters":{
"name": {
"COD_STRADA":{
"number": "COD_STRADA"
}
}
},
"spatialSelectorsConfig":{
"name":{
"ptype": "gxp_spatial_geocoding_selector",
"showSelectionSummary": false,
"multipleSelection": false,
"searchComboOutputFormat": "json",
"wfsBaseURL": "http://geoserver.comune.genova.it/geoserver/wfs",
"geocoderTypeName": "CTC:V_ASTE_STRADALI_TOPONIMO_SUB",
"geocoderTypeRecordModel":[
{
"name":"id",
"mapping":"id"
},
{
"name":"name",
"mapping":"properties.NOMEVIA"
},
{
"name":"custom",
"mapping":"properties.COD_STRADA"
},
{
"name":"geometry",
"mapping":"geometry"
}
],
"geocoderTypeSortBy":null,
"geocoderTypeQueriableAttributes":[
"COD_STRADA", "NOMEVIA"
],
"spatialOutputCRS": "EPSG:3003",
"geocoderTypePageSize": 10,
"zoomToCurrentExtent": false
},
"number":{
"ptype": "gxp_spatial_geocoding_selector",
"showSelectionSummary": false,
"multipleSelection": false,
"searchComboOutputFormat": "json",
"wfsBaseURL": "http://geoserver.comune.genova.it/geoserver/wfs",
"geocoderTypeName": "SITGEO:CIVICI_COD_TOPON_SUB",
"geocoderTypeRecordModel":[
{
"name":"id",
"mapping":"id"
},
{
"name":"name",
"mapping":"properties.COD_STRADA"
},
{
"name":"custom",
"mapping":"properties.COD_TOPON"
},
{
"name":"geometry",
"mapping":"geometry"
}
],
"geocoderTypeSortBy":null,
"geocoderTypeQueriableAttributes":[
"COD_STRADA", "COD_TOPON"
],
"spatialOutputCRS": "EPSG:3003",
"geocoderTypePageSize": 10,
"zoomToCurrentExtent": false
}
}
}
You need to know the WFS feature model and include it on the spatial selectors and the crossParameters
.
In this case we use two WFS combo, one for the street name:
...
"name":{
"ptype": "gxp_spatial_geocoding_selector",
"showSelectionSummary": false,
"multipleSelection": false,
"searchComboOutputFormat": "json",
"wfsBaseURL": "http://geoserver.comune.genova.it/geoserver/wfs",
"geocoderTypeName": "CTC:V_ASTE_STRADALI_TOPONIMO_SUB",
"geocoderTypeRecordModel":[
{
"name":"id",
"mapping":"id"
},
{
"name":"name",
"mapping":"properties.NOMEVIA"
},
{
"name":"custom",
"mapping":"properties.COD_STRADA"
},
{
"name":"geometry",
"mapping":"geometry"
}
],
"geocoderTypeSortBy":null,
"geocoderTypeQueriableAttributes":[
"COD_STRADA", "NOMEVIA"
],
"spatialOutputCRS": "EPSG:3003",
"geocoderTypePageSize": 10,
"zoomToCurrentExtent": false
}
...
And another one for the street number:
...
"number":{
"ptype": "gxp_spatial_geocoding_selector",
"showSelectionSummary": false,
"multipleSelection": false,
"searchComboOutputFormat": "json",
"wfsBaseURL": "http://geoserver.comune.genova.it/geoserver/wfs",
"geocoderTypeName": "SITGEO:CIVICI_COD_TOPON_SUB",
"geocoderTypeRecordModel":[
{
"name":"id",
"mapping":"id"
},
{
"name":"name",
"mapping":"properties.COD_STRADA"
},
{
"name":"custom",
"mapping":"properties.COD_TOPON"
},
{
"name":"geometry",
"mapping":"geometry"
}
],
"geocoderTypeSortBy":null,
"geocoderTypeQueriableAttributes":[
"COD_STRADA", "COD_TOPON"
],
"spatialOutputCRS": "EPSG:3003",
"geocoderTypePageSize": 10,
"zoomToCurrentExtent": false
}
...
And I link the first combo with the second one the COD_STRADA
parameter.
...
"crossParameters":{
"name": {
"COD_STRADA":{
"number": "COD_STRADA"
}
}
}
...
Class/Interface Name
/** api: constructor
* .. class:: Geocoder(config)
*
* Geocoder based on spatial selectors
*/
gxp.plugins.spatialselector.Geocoder = Ext.extend(gxp.plugins.Tool, {
/* ptype = gxp_spatial_selector_geocoder */
ptype : 'gxp_spatial_selector_geocoder',
/** api: config[layoutConfig]
* ``Object``
* Configuration for the output layout.
*/
/** api: config[crossParameters]
* ``Object``
* Configuration for crossParameters by selectors. When you select a result for the first combo,
* it will enable the second or more comboboxes and copy the parameter configured in the second one
*/
crossParameters:{},
/** api: config[searchBtnCls]
* ``String``
* Icon cls for the search button.
*/
searchBtnCls: "gxp-icon-find",
/** api: config[resetBtnCls]
* ``String``
* Icon cls for the search button.
*/
resetBtnCls: "cancel",
/** i18n **/
/** api: config[titleText]
* ``String``
* Title for the output (i18n).
*/
titleText: "Geocoder",
/** api: config[searchText]
* ``String``
* Search text (i18n).
*/
searchText: "Search",
/** api: config[searchTpText]
* ``String``
* Search tooltip text (i18n).
*/
searchTpText: "Search selected location and zoom in on map",
/** api: config[resetText]
* ``String``
* Reset text (i18n).
*/
resetText: "Reset",
/** api: config[resetText]
* ``String``
* Reset text (i18n).
*/
resetTpText: "Reset location search",
/** api: config[translatedKeys]
* ``String``
* Translated keys for spatial selectors (i18n).
*/
translatedKeys: {
"name": "Street",
"number": "Number"
},
/** EoF i18n **/
/** api: method[constructor]
* Init spatialSelectors .
*/
constructor : function(config) {
// default layout configuration
this.layoutConfig = {
xtype: "panel",
layout: "form",
defaults:{
layout:"fieldfet"
}
};
// Apply config
Ext.apply(this, config);
return gxp.plugins.spatialselector.SpatialSelector.superclass.constructor.call(this, arguments);
},
/** api: method[addOutput]
*/
addOutput: function() {
// initialize spatial selectors
this.spatialSelectors = {};
this.spatialSelectorsItems = [];
if(this.spatialSelectorsConfig){
for (var key in this.spatialSelectorsConfig){
var spConfig = this.spatialSelectorsConfig[key];
spConfig.target = this.target;
// Add i18n support by spatial selector
if(this.translatedKeys[key]){
spConfig["name"] = this.translatedKeys[key];
}
if(spConfig.id
&& this.target
&& this.target.tools
&& this.target.tools[spConfig.id]){
this.spatialSelectors[key] = this.target.tools[spConfig.id];
}else{
var plugin = Ext.ComponentMgr.createPlugin(spConfig);
if(this.target
&& this.target.tools){
this.target.tools[spConfig.id] = plugin;
}
this.spatialSelectors[key] = plugin;
var selectorItem = plugin.getSelectionMethodItem();
selectorItem.value = key;
this.spatialSelectorsItems.push(selectorItem);
}
}
}
// prepare layout
var layout = {};
Ext.apply(layout, this.layoutConfig);
if(!layout.title){
layout.title = this.titleText;
}
// initialize layout
layout.items = [];
if(this.spatialSelectors){
for (var key in this.spatialSelectors){
var output = this.spatialSelectors[key].addOutput();
output.on("geometrySelect", this.onGeometrySelect, {scope:this, type: key});
if(output){
if(!this.crossParameters[key]){
output.setDisabled(true);
}
layout.items.push(output);
}
}
}
layout.buttonAlign = "right";
layout.bbar = [
"->",
{
xtype : "button",
text : this.searchText,
tooltip : this.searchTpText,
iconCls: this.searchBtnCls,
scope : this,
handler : this.search
},{
xtype : "button",
text : this.resetText,
tooltip : this.resetTpText,
iconCls: this.resetBtnCls,
scope : this,
handler : this.reset
}];
var output = gxp.plugins.spatialselector.Geocoder.superclass.addOutput.call(this, layout);
return output;
},
onGeometrySelect:function(geometry){
var params = this;
var me = params.scope;
var type = params.type;
var selected = me.spatialSelectors[type].wfsComboBox.getValue();
var store = me.spatialSelectors[type].wfsComboBox.store;
var records = store.getRange();
var size = store.getCount();
var recordData = null;
for(var i = 0; i < size; i++){
var record = records[i];
if (record && record.data.name == selected) {
recordData = record;
break;
}
}
var selectedProperties = recordData.json.properties;
if(me.crossParameters[type]){
for(var parameter in me.crossParameters[type]){
var value = selectedProperties[parameter];
for(var targetCombo in me.crossParameters[type][parameter]){
// enable combo
me.spatialSelectors[targetCombo].output.setDisabled(false);
var comboBox = me.spatialSelectors[targetCombo].wfsComboBox;
// Apply filter (only one at time)
if(!comboBox.vendorParams){
comboBox.vendorParams = {};
}
var cql_filter = me.crossParameters[type][parameter][targetCombo] + " = " + value;
Ext.apply(comboBox.vendorParams,{
cql_filter: cql_filter
});
}
}
}
me.geometry = geometry;
},
/** api: method[reset]
* Search action.
*/
search: function(){
var geometry = this.geometry;
if (geometry && geometry.getBounds) {
var dataExtent = geometry.getBounds();
this.target.mapPanel.map.zoomToExtent(dataExtent, closest=false);
}
},
/** api: method[reset]
* Reset the state of the Geocoder.
*/
reset: function(){
this.geometry = null;
if(this.spatialSelectors){
for (var key in this.spatialSelectors){
this.spatialSelectors[key].wfsComboBox.reset();
this.spatialSelectors[key].reset();
if(!this.crossParameters[key]){
this.spatialSelectors[key].output.setDisabled(true);
}
}
}
}
});
The most important configurations are:
- spatialSelectorsConfig: Spatial selector plugins configurations
- crossParameters: Configuration for crossParameters by selectors. When you select a result for the first combo, it will enable the second or more comboboxes and copy the parameter configured in the second one
You can extend it as you want, for example, you can concatenate more than two WFS layers.
Feedback
This section should contain feedback provided by members who may have a problem with the proposal.
Backwards Compatibility
The new code will be compatible with GXP and MapPanel. Should be therefore backward compatible with all versions of MapStore.
Voting
Alejandro Díaz: +1 Mauro Bartolomeoli: +1