-
Notifications
You must be signed in to change notification settings - Fork 4
GSIP 81 GetLegendGraphic as text (JSON)
To compose the legend on the client side making it interactive localized and styled in the way the web-styler (or the map component) wants.
Carlo Cancellieri
master then backport to 2.2.3 or 2.2.4
Choose one of: Under Discussion, In Progress, Completed, Rejected, Deferred
GeoServer should be able to create an interactive and configurable legend on the client side.
Some of the advantages which can be achieved are:
- Get filters (intervals and comparators), colors or graphics on the client side as individual objects to apply css, layout and modify the shape of the legend accordingly with the site style and layout.
- Get labels as texts (so they can be localized client side).
- In combination with the json outputFormat getFeatureInfo (or imageMap ) can be used to highlight the element of the legend representing that value into the map and vice versa (see here for an example).
NOTE that most of the parts of this response handler encoder can be reused to serialize the styles in json format (which currently is still missing).
{quote}{}NOTE The getLegendGraphic does not implements all the output format supported by the getMap as specified here:http://docs.geoserver.org/latest/en/user/services/wms/getlegendgraphic/legendgraphic.html#get-legend-graphic
{quote}Create a response handler to replay to getLegendGraphic requests in text json format.
My intension is to provide a textual representation of the rules for a target layer/style (in an OpenLayer like format) including:
- names, title, …
- filter (as structured object)
- symbolizers (also with graphic objects)
Embedded URL
advantages:
- Small and quick response body.
- You can selectively download only really needed graphical resources.
disadvantages:
- To show the graphic resources into the legend, multiple getLegendGraphic (filtered) requests to the GeoServer should be performed.
Embedded graphic resource
advantages:
- Get all the legend resources in one shot.
- No other request are needed.
disadvantages:
- For some style you may have many external resources in some case this may result in a really big response.
So depending on the style you are using and the legend you want to generate, (sometime you only want to show the first (lower?) and the last (higher?) values), users may prefer the embedded url or the embedded graphic resource version.
The graphic object will be serialized in base64 so the client can use it directly as image. fe:
<img alt="Embedded Image" width="80" height="31"
src="%3D%3D" />
Will result in:
A good alternative (for big images) is to use the RULE parameter of the getLegendGraphic (image output format) to renderize and return only one graphic object at time.
*RULE* Optional Rule of style to produce legend graphic for, if applicable. In the case that a style has multiple rules but no specific rule is selected, then the map server is obligated to produce a graphic that is representative of all of the rules of the style.
The rules of the dynamic symbolizers will be resolved for each feature and the list of obtained external graphic will be serialized as a list of rules with the name initialized to the value of the resolved attribute.
For example:
<FeatureTypeStyle>
<Rule>
<Name>Flags</Name>
<Title>USA state flags</Title>
<PointSymbolizer>
<Graphic>
<ExternalGraphic>
<OnlineResource xlink:type="simple"
xlink:href="http://www.usautoparts.net/bmw/images/states/tn_${strToLowerCase(STATE_ABBR)}.jpg" />
<Format>image/gif</Format>
</ExternalGraphic>
</Graphic>
</PointSymbolizer>
</Rule>
</FeatureTypeStyle>
Will be resolved as:
<FeatureTypeStyle>
<Rule>
<Name>alaska</Name>
<Title>USA state flags</Title>
<PointSymbolizer>
<Graphic>
<ExternalGraphic>
<OnlineResource xlink:type="simple"
xlink:href="http://www.usautoparts.net/bmw/images/states/tn_alaska.jpg" />
<Format>image/gif</Format>
</ExternalGraphic>
</Graphic>
</PointSymbolizer>
</Rule>
...
<Rule>
<Name>wyoming</Name>
<Title>USA state flags</Title>
<PointSymbolizer>
<Graphic>
<ExternalGraphic>
<OnlineResource xlink:type="simple"
xlink:href="http://www.usautoparts.net/bmw/images/states/tn_wyoming.jpg" />
<Format>image/gif</Format>
</ExternalGraphic>
</Graphic>
</PointSymbolizer>
</Rule>
</FeatureTypeStyle>
Then serialized as json.
If the OnlineResource is a relative path we have 2 ways to proceed:
-
generate a base64 encoded string to encode and send the image.
-
use getLegendGraphic with image output format to be able to serve images from the FileSystem (selecting them using the RULE parameter and probably the CQL_FIlter to select the feature by FeatureID f.e.). I think this (cql_filter) can be another good improvement if it is not supported.
As said the graphical objects can be:
- serialized as is (json style representation)
This will minimize the data transfer and the load on the target geoserver delegating render operation to the client library this make sense for standard Mark objects (ttf://, shape://, …)
-
rendered all graphic objects as embedded images into the response encoded as base64 string (json with graphic embedded).
-
provide links to external graphic using geoserver url (s):
NOTE: For dynamic symbolizers we need the cql_filter improvement
{quote}{}Note: the below examples are used only to discuss the proposal, this is not the final representation of the getLegendGraphic as text. {quote}Using this call:
You will get something like:
{
getLegendGraphic:[
{
description:null,
title:null,
name:null,
symbolyzers:[
{
name:null,
description:null,
title:null,
type:“RasterSymbolizer”,
colormap:{
entries:[
{
label:“nodata”,
opacity:“0.0”,
quantity:“–500”,
color:“\#000000”
},
{
label:“values”,
opacity:null,
quantity:“0”,
color:“\#AAFFAA”
},
{
label:null,
opacity:null,
quantity:“1000”,
color:“\#00FF00”
},
{
label:“values”,
opacity:null,
quantity:“1200”,
color:“\#FFFF00”
},
{
label:“values”,
opacity:null,
quantity:“1400”,
color:“\#FF7F00”
},
{
label:“values”,
opacity:null,
quantity:“1600”,
color:“\#BF7F3F”
},
{
label:“values”,
opacity:null,
quantity:“2000”,
color:“\#000000”
}
]
},
geometry:“geom”,
geometryPropertyName:“geom”
}
]
}
]
}
This is the matching style:
<?xml version="1.0" encoding="ISO-8859-1"?>
<StyledLayerDescriptor version="1.0.0" xmlns="http://www.opengis.net/sld" xmlns:ogc="http://www.opengis.net/ogc"
xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.opengis.net/sld http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd">
<NamedLayer>
<Name>gtopo</Name>
<UserStyle>
<Name>dem</Name>
<Title>
Simple DEM style
</Title>
<Abstract>Classic elevation color progression</Abstract>
<FeatureTypeStyle>
<Rule>
<RasterSymbolizer>
<Opacity>1.0</Opacity>
<ColorMap>
<ColorMapEntry color="#000000" quantity="-500" label="nodata" opacity="0.0" />
<ColorMapEntry color="#AAFFAA" quantity="0" label="values" />
<ColorMapEntry color="#00FF00" quantity="1000"/>
<ColorMapEntry color="#FFFF00" quantity="1200" label="values" />
<ColorMapEntry color="#FF7F00" quantity="1400" label="values" />
<ColorMapEntry color="#BF7F3F" quantity="1600" label="values" />
<ColorMapEntry color="#000000" quantity="2000" label="values" />
</ColorMap>
</RasterSymbolizer>
</Rule>
</FeatureTypeStyle>
</UserStyle>
</NamedLayer>
</StyledLayerDescriptor>
{quote}{}NOTE take a look to the filters shown here which should result in an OpenLayer compatible representation. {quote}
{
getLegendGraphic:[
{
description:null,
title:"Default",
name:"Default",
symbolyzers:[
{
name:"Default",
description:null,
title:"Default",
Polygon:{
fill:true,
opacity:0,
fillColor:"#ffffff",
strokeColor:"#c0c0c0",
strokeDashArray:"null",
strokeDashOffset:0
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class0",
name:"class0",
filter:{
type:"<=",
property:"total_usd_percapita",
value:"env([c0_upper], [1])",
matchCase:false
},
symbolyzers:[
{
name:"class0",
description:"...",
title:"class0",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c0_fill], [#00ff00])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class1",
name:"class1",
filter:{
type:"&&",
filters:[
{
type:"<=",
property:"total_usd_percapita",
value:"env([c1_upper], [2])",
matchCase:false
},
{
type:">",
property:"total_usd_percapita",
value:"env([c1_lower], [1])",
matchCase:true
}
]
},
symbolyzers:[
{
name:"class1",
description:"...",
title:"class1",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c1_fill], [#00ff00])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class2",
name:"class2",
filter:{
type:"&&",
filters:[
{
type:"<=",
property:"total_usd_percapita",
value:"env([c2_upper], [4])",
matchCase:false
},
{
type:">",
property:"total_usd_percapita",
value:"env([c2_lower], [2])",
matchCase:true
}
]
},
symbolyzers:[
{
name:"class2",
description:"...",
title:"class2",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c2_fill], [#00ff00])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class3",
name:"class3",
filter:{
type:"&&",
filters:[
{
type:"<=",
property:"total_usd_percapita",
value:"env([c3_upper], [5])",
matchCase:false
},
{
type:">",
property:"total_usd_percapita",
value:"env([c3_lower], [4])",
matchCase:true
}
]
},
symbolyzers:[
{
name:"class3",
description:"...",
title:"class3",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c3_fill], [#00ff00])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class4",
name:"class4",
filter:{
type:">",
property:"total_usd_percapita",
value:"env([c4_lower], [5])",
matchCase:true
},
symbolyzers:[
{
name:"class4",
description:"...",
title:"class4",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c4_fill], [#00ff00])"
},
geometry:null,
geometryPropertyName:null
}
]
}
]
}
{
getLegendGraphic:[
{
description:"...",
title:"no_data",
name:"no_data",
filter:{
type:"==",
property:"fk_d_area",
value:0,
matchCase:true
},
symbolyzers:[
{
name:"no_data",
description:"...",
title:"no_data",
Polygon:{
fill:true,
opacity:1,
fillColor:"#cccccc"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class0",
name:"class0",
filter:{
type:"&&",
filters:[
{
type:"!=",
property:"fk_d_area",
value:0,
matchCase:true
},
{
type:"<",
property:"at_value",
value:"env([c0_upper], [0])",
matchCase:false
}
]
},
symbolyzers:[
{
name:"class0",
description:"...",
title:"class0",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c0_fill], [#ffffff])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class1",
name:"class1",
filter:{
type:"&&",
filters:[
{
type:"!=",
property:"fk_d_area",
value:0,
matchCase:true
},
{
type:">=",
property:"at_value",
value:"env([c1_lower], [0])",
matchCase:false
},
{
type:"<",
property:"at_value",
value:"env([c1_upper], [500])",
matchCase:false
}
]
},
symbolyzers:[
{
name:"class1",
description:"...",
title:"class1",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c1_fill], [#cccccc])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class2",
name:"class2",
filter:{
type:"&&",
filters:[
{
type:"!=",
property:"fk_d_area",
value:0,
matchCase:true
},
{
type:">=",
property:"at_value",
value:"env([c2_lower], [500])",
matchCase:false
},
{
type:"<",
property:"at_value",
value:"env([c2_upper], [1000])",
matchCase:false
}
]
},
symbolyzers:[
{
name:"class2",
description:"...",
title:"class2",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c2_fill], [#BEDED8])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class3",
name:"class3",
filter:{
type:"&&",
filters:[
{
type:"!=",
property:"fk_d_area",
value:0,
matchCase:true
},
{
type:">=",
property:"at_value",
value:"env([c3_lower], [1000])",
matchCase:false
},
{
type:"<",
property:"at_value",
value:"env([c3_upper], [1700])",
matchCase:false
}
]
},
symbolyzers:[
{
name:"class3",
description:"...",
title:"class3",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c3_fill], [#6BAFB6])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class4",
name:"class4",
filter:{
type:"&&",
filters:[
{
type:"!=",
property:"fk_d_area",
value:0,
matchCase:true
},
{
type:">=",
property:"at_value",
value:"env([c4_lower], [1700])",
matchCase:false
},
{
type:"<",
property:"at_value",
value:"env([c4_upper], [6000])",
matchCase:false
}
]
},
symbolyzers:[
{
name:"class4",
description:"...",
title:"class4",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c4_fill], [#2C80B8])"
},
geometry:null,
geometryPropertyName:null
}
]
},
{
description:"...",
title:"class5",
name:"class5",
filter:{
type:"&&",
filters:[
{
type:"!=",
property:"fk_d_area",
value:0,
matchCase:true
},
{
type:">=",
property:"at_value",
value:"env([c5_lower], [6000])",
matchCase:false
}
]
},
symbolyzers:[
{
name:"class5",
description:"...",
title:"class5",
Polygon:{
fill:true,
opacity:1,
fillColor:"env([c5_fill], [#08559C])"
},
geometry:null,
geometryPropertyName:null
}
]
}
]
}
This section should contain feedback provided by PSC members who may have a problem with the proposal.
Regarding the outputFormat and the format parameter, currently the unique way to add as add on (without modify the current getLegendGraphic output format) is implementing another response handler and a new output format which may results in query like:
For a better integration of the getLegendGraphic I think we may define if we want to switch using different responseHandlers or continue working with outputFormats.Let’s analyse differences:
Alessio Fabiani: Andrea Aime: Ben Caradoc Davies: +0 Christian Mueller: +0 Gabriel Roldan: Jody Garnett: Jukka Rahkonen: Justin Deoliveira: Phil Scadden: +1 Simone Giannecchini:
JIRA Task Email Discussion Wiki Page