This page last changed on Apr 11, 2014 by jive.

Overview

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.

Proposed By

Carlo Cancellieri

Assigned to Release

master then backport to 2.2.3 or 2.2.4

State

Choose one of: Under Discussion, In Progress, Completed, Rejected, Deferred

Motivation

GeoServer should be able to create an interactive and configurable legend on the client side.

Some of the advantages which can be achieved are:

  1. 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.
  2. Get labels as texts (so they can be localized client side).
  3. 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).

Proposal

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/get_legend_graphic/legendgraphic.html#get-legend-graphic

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:

  1. names, title, ...
  2. filter (as structured object)
  3. symbolizers (also with graphic objects)

External resources

Embedded URL

advantages:

  1. Small and quick response body.
  2. You can selectively download only really needed graphical resources.

disadvantages:

  1. To show the graphic resources into the legend, multiple getLegendGraphic (filtered) requests to the GeoServer should be performed.

Embedded graphic resource

advantages:

  1. Get all the legend resources in one shot.
  2. No other request are needed.

disadvantages:

  1. 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.
dynamic symbolizers

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.
SVG, or Mark objects, or graphic fills and graphic strokes

As said the graphical objects can be:default

  • 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):

F.e.:http://localhost:80///geoserver/wms?REQUEST=GetLegendGraphic&VERSION=....&FORMAT=image/png&LAYER=layerName&RULE=RuleName&LEGEND_OPTIONS=forceLabels:off

NOTE: For  dynamic symbolizers we need the cql_filter improvement

Examples

Note: the below examples are used only to discuss the proposal, this is not the final representation of the getLegendGraphic as text.

Using this call:

http://localhost:8080/geoserver/sf/wms?service=WMS&version=1.1.0&request=GetLegendGraphic&layer=sf:sfdem&styles=&bbox=589980.0,4913700.0,609000.0,4928010.0&width=512&height=385&srs=EPSG:26713&format=application/json&outputFormat=application/json

You will get something like:

Raster colormap

{
   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>

Filter Example (1):

NOTE take a look to the filters shown here which should result in an OpenLayer compatible representation.

{
   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
            }
         ]
      }
   ]
}

Filter Example (2):

{
   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
            }
         ]
      }
   ]
}

Feedback

This section should contain feedback provided by PSC members who may have a problem with the proposal.

Backwards Compatibility

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:

http://localhost:8080/geoserver/topp/wms?service=WMS&version=1.1.0&request=GetLegendGraphic&layer=topp:states&styles=population2&bbox=-124.73142200000001,24.955967,-66.969849,49.371735&width=780&height=330&srs=EPSG:4326&format=application/json&outputFormat=application/json

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:

Voting

Alessio Fabiani:
Andrea Aime:
Ben Caradoc-Davies: +0
Christian Mueller: +0
Gabriel Roldán:
Jody Garnett:
Jukka Rahkonen:
Justin Deoliveira:
Phil Scadden: +1
Simone Giannecchini:

Links

[JIRA Task|]
[Email Discussion|]
[Wiki Page|]


pbGS_80x31glow.png (image/png)
Document generated by Confluence on May 14, 2014 23:00