3.6. CSS Workbook Conclusion

We hope you have enjoyed this styling workshop.

Additional resources:

3.6.1. CSS Workbook Answer Key

The following questions were listed throughout the workshop as an opportunity to explore the material in greater depth. Please do your best to consider the questions in detail prior to checking here for the answer. Questions are provided to teach valuable skills, such as a chance to understand how feature type styles are used to control z-order, or where to locate information in the user manual.

3.6.1.1. SLD Generation

Answer for Challenge SLD Generation:

  1. Generate the SLD for the following CSS.

    * {
      stroke: black;
    }
    
  2. Challenge: What is unusual about the generated SLD? Can you explain why it still works as expected?

    The generated SLD does not contain any stroke properties, even though black was specified:

    <sld:LineSymbolizer>
      <sld:Stroke/>
    </sld:LineSymbolizer>
    

    SLD considers black the default stroke color for a LineSymbolizer, so no further detail was required.

3.6.1.2. Classification

Answer for Challenge Classification:

  1. Challenge: Create a new style adjust road appearance based on type.

    ../../../_images/line_type.png

    Hint: The available values are ‘Major Highway’,’Secondary Highway’,’Road’ and ‘Unknown’.

  2. Here is an example:

    [type = 'Major Highway' ] {
        stroke: #000088;
        stroke-width: 1.25;
    }
    [type = 'Secondary Highway' ]{
        stroke: #8888AA;
        stroke-width: 0.75;
    }
    [type = 'Road']{
        stroke: #888888;
        stroke-width: .75;
    }
    [type = 'Unknown' ]{
        stroke: #888888;
        stroke-width: 0.5;
    }
    * {
       stroke: #AAAAAA;
       stroke-opacity: 0.25;
       stroke-width: 0.5;
    }
    

3.6.1.3. SLD Z-Index Generation

Answer for Challenge SLD Z-Index Generation:

  1. Review the SLD generated by the z-index example.

    * {
      stroke: black, #8080E6;
      stroke-width: 5px, 3px;
      z-index: 0, 1;
    }
    
  2. Challenge: There is an interesting trick in the generated SLD, can you explain how it works?

  3. The Z-Order example produces multiple FeatureTypeSytle definitions, each acting like an “inner layer”.

    Each FeatureTypeStyle is rendered into its own raster, and the results merged in order. The legend shown in the map preview also provides a hint, as the rule from each FeatureType style is shown.

3.6.1.4. Label Shields

Answer for Challenge Label Shields:

  1. The traditional presentation of roads in the US is the use of a shield symbol, with the road number marked on top.

../../../_images/line_shield.png
  1. Challenge: Have a look at the documentation and reproduce this technique.

  2. The use of a label shield is a vendor specific capability of the GeoServer rendering engine. The tricky part of this exercise is finding the documentation online ( i.e. Styled Marks in CSS).

    * {
        stroke: black,lightgray;
        stroke-width: 3,2;
        label: [name];
        font-family: 'Ariel';
        font-size: 10;
        font-fill: black;
        shield: symbol(square);
    }
    :shield {
        fill: white;
        stroke: black;
        size: 18;
    }
    

3.6.1.5. Antialiasing

Answer for Explore Antialiasing:

  1. When we rendered our initial preview, without a stroke, thin white gaps (or slivers) are visible between our polygons.

    ../../../_images/polygon_04_preview.png

    This effect is made more pronounced by the rendering engine making use of the Java 2D sub-pixel accuracy. This technique is primarily used to prevent an aliased (stair-stepped) appearance on diagonal lines.

  2. Explore: Experiment with fill and stroke settings to eliminate slivers between polygons.

    The obvious approach works - setting both values to the same color:

    symbolizers:
    - polygon:
        stroke-color: 'lightgrey'
        stroke-width: 1
        fill-color: 'lightgrey'
    

3.6.1.6. Categorize

Answer for Explore Categorize:

  1. An exciting use of the GeoServer shape symbols is the theming by changing the size used for pattern density.

  2. Explore: Use the Categorize function to theme by datarank.

    ../../../_images/polygon_categorize.png

    Example:

    .. code-block:: css
    
       * {
         fill: symbol('shape://slash');
         fill-size: [
            Categorize(datarank,
             4, 4,
             5, 6,
             8, 10,
            10)
         ];
         stroke: black;
       }
       :fill {
         stroke: darkgray;
       }
    

3.6.1.7. Halo

Answer for Challenge Halo:

  1. The halo example used the fill color and opacity for a muted halo, while this improved readability it did not bring attention to our labels.

    A common design choice for emphasis is to outline the text in a contrasting color.

  2. Challenge: Produce a map that uses a white halo around black text.

    Here is an example:

    * {  stroke: gray;
         fill: #7EB5D3;
         label: [name];
         label-anchor: 0.5 0.5;
         font-fill: black;
         font-family: "Arial";
         font-size: 14;
         halo-radius: 1;
         halo-color: white;
       }
    

3.6.1.8. Theming using Multiple Attributes

Answer for Challenge Theming using Multiple Attributes:

  1. A powerful tool is theming using multiple attributes. This is an important concept allowing map readers to perform “integration by eyeball” (detecting correlations between attribute values information).

  2. Challenge: Combine the mapcolor9 and datarank examples to reproduce the following map.

    ../../../_images/polygon_multitheme.png

    This should be a cut and paste using the recode example, and categorize examples already provided.

    * {
       fill: [
        recode(mapcolor9,
          1,'#8dd3c7', 2,'#ffffb3', 3,'#bebada',
          4,'#fb8072', 5,'#80b1d3', 6,'#fdb462',
          7,'#b3de69', 8,'#fccde5', 9,'#d9d9d9')
       ], symbol('shape://slash');
    
       fill-size: '',[
          Categorize(datarank,
           6, 4,
           8, 6,
          10, 10,
          12)
       ];
       stroke: black;
    }
    :fill {
       stroke: black;
    }
    

3.6.1.9. Use of Use of Z-Index

Answer for Challenge Use of Z-Index:

  1. Earlier we looked at using z-index to simulate line string casing. The line work was drawn twice, once with thick line, and then a second time with a thinner line. The resulting effect is similar to text halos - providing breathing space around complex line work allowing it to stand out.

  2. Challenge: Use what you know of LineString z-index to reproduce the following map:

    ../../../_images/polygon_zorder.png

    This is a tricky challenge. While it is easy enough to introduce z-index to control stroke what is not immediately obvious is that z-order also controls fill order. The previous examples illustrate how to introduce z-order, some thought is required to untangle fill and stroke z-order (dummy stroke definitions need to be introduced using empty commas).

    * {
      fill: lightgray, symbol('shape://slash');
      fill-size: 8px;
      stroke: '','',lightgray, black;
      stroke-width: '','',6,1.5;
      z-index: 1,2,3,4;
    }
    :fill {
      stroke: black;
      stroke-width: 0.75;
    }
    

    The included legend should be a large clue about what is going on.

3.6.1.10. Geometry Location

Answer for Challenge Geometry Location:

  1. The mark property can be used to render any geometry content.

  2. Challenge: Try this yourself by rendering a polygon layer using a mark property.

    This can be done one of two ways:

    • Changing the association of a polygon layer, such as ne:states_provinces_shp to point_example and using the layer preview page.

    • Changing the Layer Preview tab to a polygon layer, such as ne:states_provinces_shp.

    The important thing to notice is that the centroid of each polygon is used as a point location.

3.6.1.11. Dynamic Symbolization

Answer for Explore Dynamic Symbolization:

  1. SLD Mark and ExternalGraphic provide an opportunity for dynamic symbolization.

    This is accomplished by embedding a small CQL expression in the string passed to symbol or url. This sub-expression is isolated with ${ } as shown:

    - point:
        symbols:
        - mark:
            shape: ${if_then_else(equalTo(FEATURECLA,'Admin-0 capital'),'star','circle')}
    
  2. Challenge: Use this approach to rewrite the Dynamic Styling example.

    Example available here point_example.css

3.6.1.12. Layer Group

Answer for Challenge Layer Group:

  1. Use a Layer Group to explore how symbology works together to form a map.

    • ne:NE1

    • ne:states_provincces_shp

    • ne:populated_places

  2. This background is relatively busy and care must be taken to ensure both symbols and labels are clearly visible.

  3. Challenge: Do your best to style populated_places over this busy background.

    Here is an example with labels for inspiration:

    ../../../_images/point_challenge_1.png

    This should be an opportunity to revisit label halo settings from Polygons.

    * {
       mark-size: [5+((10-SCALERANK)/3)];
    
       font-fill: black;
       font-family: "Arial";
       font-size: 10;
    
       label-anchor: 0.5 1;
       label-offset: 0 [-12+SCALERANK];
    
       halo-radius: 2;
       halo-color: lightgray;
       halo-opacity:0.7;
    
       mark-label-obstacle: true;
       label-max-displacement: 90;
       label-priority: [0 - LABELRANK];
    }
    :symbol {
      fill: black;
      stroke: white;
      stroke-opacity:0.75;
    }
    

    Using a lightgray halo, 0.7 opacity and radius 2 fades out the complexity immediately surrounding the label text improving legibility.

3.6.1.13. Contrast Enhancement

Discussion for Explore Contrast Enhancement:

  1. A special effect that is effective with grayscale information is automatic contrast adjustment.

  2. Make use of a simple contrast enhancement with usgs:dem:

    * {
        raster-channels: auto;
        raster-contrast-enhancement: normalize;
    }
    
  3. Can you explain what happens when zoom in to only show a land area (as indicated with the bounding box below)?

    ../../../_images/raster_contrast_1.png

    What happens is insanity, normalize stretches the palette of the output image to use the full dynamic range. As long as we have ocean on the screen (with value 0) the land area was shown with roughly the same presentation.

    ../../../_images/raster_contrast_2.png

    Once we zoom in to show only a land area, the lowest point on the screen (say 100) becomes the new black, radically altering what is displayed on the screen.

3.6.1.14. Intervals

Answer for Challenge Intervals:

  1. The color-map type property dictates how the values are used to generate a resulting color.

    • ramp is used for quantitative data, providing a smooth interpolation between the provided color values.

    • intervals provides categorization for quantitative data, assigning each range of values a solid color.

    • values is used for qualitative data, each value is required to have a color-map entry or it will not be displayed.

  2. Challenge: Update your DEM example to use intervals for presentation. What are the advantages of using this approach for elevation data?

    By using intervals it becomes very clear how relatively flat most of the continent is. The ramp presentation provided lots of fascinating detail which distracted from this fact.

    ../../../_images/raster_interval.png

    Here is style for you to cut and paste:

    * {
      raster-channels: auto;
      raster-color-map:
         color-map-entry(#014636,   0,0)
         color-map-entry(#014636,   1)
         color-map-entry(#016c59, 500)
         color-map-entry(#02818a,1000)
         color-map-entry(#3690c0,1500)
         color-map-entry(#67a9cf,2000)
         color-map-entry(#a6bddb,2500)
         color-map-entry(#d0d1e6,3000)
         color-map-entry(#ece2f0,3500)
         color-map-entry(#fff7fb,4000);
      raster-color-map-type: intervals;
    }
    

3.6.1.15. Clear Digital Elevation Model Presentation

Answer for Challenge Clear Digital Elevation Model Presentation:

  1. Now that you have seen the data on screen and have a better understanding how would you modify our initial gray-scale example?

  2. Challenge: Use what you have learned to present the usgs:dem clearly.

    ../../../_images/raster_grayscale.png

    The original was a dark mess. Consider making use of mid-tones (or adopting a sequential palette from color brewer) in order to fix this. In the following example the ocean has been left dark, allowing the mountains stand out more.

    * {
      raster-channels: auto;
      raster-color-map: color-map-entry(#000000, 0)
                        color-map-entry(#444444, 1)
                        color-map-entry(#FFFFFF, 3000);
    }
    

3.6.1.16. Raster Opacity

Discussion for Challenge Clear Digital Elevation Model Presentation:

  1. There is a quick way to make raster data transparent, raster opacity property works in the same fashion as with vector data. The raster as a whole will be drawn partially transparent allow content from other layers to provide context.

  2. Challenge: Can you think of an example where this would be useful?

    This is difficult as raster data is usually provided for use as a basemap, with layers being drawn over top.

    The most obvious example here is the display of weather systems, or model output such as fire danger. By drawing the raster with some transparency, the landmass can be shown for context.