Filtering in SVG
Filtering can be used to create special effects that really make an image pop. Filters manipulate raster images instead of vector graphics. So, everything in this section talks about pixels and images instead of vectors. SVG viewer support for filters is often weak but the major browsers have been improving.
The filter element is a container for one or more filter effects. The effects are what describe what the filter does. The filter is used by referencing it using the filter attribute that all graphic elements and container elements have. Thus, filters need to have an id attribute. With a basic filter, the other attributes can be left alone.
Adding a filter means that the contained vector graphics are no longer directly rendered. The rasterized image of the vector graphics can be used as input to the filter effects. There is no rule that the contained graphics must be used at all. So, a filter with no effects would just hide what it is attached to.
<defs> <filter id="myPrettyFilter"> <!-- fe elements go here--> </filter> </defs> <rect x="10" y="10" width="200" height="200" filter="url(#myPrettyFilter)" />
Attribute |
Initial |
Explanation |
id |
The unique id required to reference this filter. |
|
filterUnits |
objectBoundingBox |
'userSpaceOnUse' or 'objectBoundingBox'. The second value makes units of x, y, width, height a fraction (or percentage) of the object bounding box which uses the filter. (default: objectBoundingBox) |
x, y |
-10% |
The clipping plane of the filter. |
width, height |
120% |
The clipping plane of the filter. |
primitiveUnits |
userSpaceOnUse |
'userSpaceOnUse' or 'objectBoundingBox'. Alters the units that filter effects express themselves in. The second value makes units of x, y, width, height a fraction (or percentage) of the filter's bounding box. |
filterRes |
Optional: The width and height in pixels of any intermediate images created to display the filter. Values separated by whitespace or a comma. By default, the SVG viewer will pick values appropriate for the device rendering the file. |
Filter Bounds
When a graphic element or container element uses a filter, an offscreen raster image is created. The size of this image is controlled by the filter dimension attributes. Each effect renders within the box established by the filter. That means filter effects and the image of the vector graphics the filter is attached to are cropped to that box. Finally, the resulting raster image is layered like any other SVG element.
<defs> <filter id="passThrough" > <!-- x: -10%, y: -10%, width: 120%, height: 120% --> <feOffset in="SourceGraphic" /> </filter> <filter id="ptOffset" > <!-- Move 20% over --> <feOffset in="SourceGraphic" dx="20" /> </filter> <filter id="ptOffsetWider" width="140%"> <feOffset in="SourceGraphic" dx="20" /> </filter> </defs> ... <g stroke="skyblue" stroke-width="2" fill="none" > <circle cx="70" cy="80" r="50" /> <circle cx="230" cy="80" r="50" filter="url(#passThrough)" /> <circle cx="390" cy="80" r="50" filter="url(#ptOffset)" /> <circle cx="550" cy="80" r="50" filter="url(#ptOffsetWider)" /> </g>
<defs> <filter id="userSpaceDemo" filterUnits="userSpaceOnUse" x="100" y="50" width="200" height="100"> <feOffset in="SourceGraphic" /> </filter> </defs> ... <g stroke="skyblue" stroke-width="2" fill="green" > <circle cx="100" cy="60" r="50" opacity=".1" /> <circle cx="100" cy="60" r="50" filter="url(#userSpaceDemo)" /> </g>
Putting it Together: Drop Shadow
One common effect that seems to be missing from the list of SVG filter effects is a drop shadow. That is a pretty common (and overused) effect. The secret is that with the other filters and smart use of named results you can build a drop shadow yourself. This example could be extended with a feColorMatrix effect to color the shadow.
<defs> <filter id="dropShadow"> <!-- Blur the source alpha to make a nice shadow. --> <feGaussianBlur in="SourceAlpha" stdDeviation="3" result="blur"/> <!-- Move the shadow over a bit. --> <feOffset in="blur" dx="3" dy="3" result="offsetBlur"/> <!-- Put everything together. Blur then graphic. --> <feMerge> <feMergeNode in="offsetBlur"/> <feMergeNode in="SourceGraphic"/> </feMerge> </filter> </defs> <text x="320" y="35" text-anchor="middle" fill="green" stroke="black" font-size="40" filter="url(#dropShadow)"> Look at my beautiful shadow.</text>