library(minisvg)

Example images

Using the example image from the MDN site. For these vignettes, I’m going to use an inline version of the image for future-proofing.

inline_img <- paste0("data:image/png;base64,", base64enc::base64encode("./images/mdn_logo.png"))

feBlend

The <feBlend> SVG filter primitive composes two objects together ruled by a certain blending mode. This is similar to what is known from image editing software when blending two layers. The mode is defined by the mode attribute.

Example from MDN docs.

doc <- svg_doc(
  width = 200, height = 200,
  stag$defs(
    stag$filter(
      id="spotlight",
      stag$feFlood(
        x=0, y=0,
        result        = "floodFill",
        width         = "100%",
        height        = "100%",
        flood_color   = "green",
        flood_opacity = "1"
      ),
      stag$feBlend(
        in_ = "SourceGraphic", in2="floodFill", mode="multiply"
      )
    )
  ),

  stag$image(
    xlink_href = inline_img,
    x="10%", y="10%", width="80%", height="80%",
    style="filter:url(#spotlight);"
  )
)

Show SVG text (click to open)

#> <?xml version="1.0" encoding="UTF-8"?>
#> <svg width="200" height="200" viewBox="0 0 200 200" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
#>   <defs>
#>     <filter id="spotlight">
#>       <feFlood flood-color="green" flood-opacity="1" result="floodFill" x="0" y="0" width="100%" height="100%" />
#>       <feBlend in="SourceGraphic" in2="floodFill" mode="multiply" />
#>     </filter>
#>   </defs>
#>   <image x="10%" y="10%" width="80%" height="80%" xlink:href="" style="filter:url(#spotlight);" />
#> </svg>


feColorMatrix

The <feColorMatrix> SVG filter element changes colors based on a transformation matrix. Every pixel’s color value (represented by an [R,G,B,A] vector) is matrix multiplied to create a new color.

Example from MDN docs.

doc <- svg_doc(
  width="40%", height="40%", viewBox="0 0 150 250",
  preserveAspectRatio = "xMidYMid meet",

  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  # 3 overlapping circles
  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  stag$defs(
    stag$g(
      id="circles",
      stag$circle(cx=30, cy=30, r=20, fill="blue" , fill_opacity="0.5"),
      stag$circle(cx=20, cy=50, r=20, fill="green", fill_opacity="0.5"),
      stag$circle(cx=40, cy=50, r=20, fill="red"  , fill_opacity="0.5")
    )
  ),

  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  # Reference rendering of the circles
  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  stag$use(href='#circles'),
  stag$text(x=70, y=50, "Reference"),


  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  # Applying a colour transformation matrix to these circles
  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  stag$filter(
    id="colorMeMatrix",
    stag$feColorMatrix(
      in_    = "SourceGraphic",
      type   = "matrix",         # matrix | saturate | hueRotate | luminanceToAlpha
      values = "0 0 0 0 0   1 1 1 1 0   0 0 0 0 0   0 0 0 1 0"
    )
  ),
  stag$use(href='#circles', transform="translate(0 70)", filter="url(#colorMeMatrix)"),
  stag$text(x=70, y=120, "Matrix"),

  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  # Desaturate the colour of the circles
  #~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  stag$filter(
    id = "colorMeSaturate",
    stag$feColorMatrix(
      in_    = "SourceGraphic",
      type   = "saturate",
      values = "0.2"
    )
  ),
  stag$use(href='#circles', transform="translate(0 140)", filter="url(#colorMeSaturate)"),
  stag$text(x=70, y=190, "Saturate")
)

Show SVG text (click to open)

#> <?xml version="1.0" encoding="UTF-8"?>
#> <svg width="40%" height="40%" viewBox="0 0 150 250" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
#>   <defs>
#>     <g id="circles">
#>       <circle cx="30" cy="30" r="20" fill="blue" fill-opacity="0.5" />
#>       <circle cx="20" cy="50" r="20" fill="green" fill-opacity="0.5" />
#>       <circle cx="40" cy="50" r="20" fill="red" fill-opacity="0.5" />
#>     </g>
#>   </defs>
#>   <use href="#circles" />
#>   <text x="70" y="50">
#>     Reference
#>   </text>
#>   <filter id="colorMeMatrix">
#>     <feColorMatrix in="SourceGraphic" type="matrix" values="0 0 0 0 0   1 1 1 1 0   0 0 0 0 0   0 0 0 1 0" />
#>   </filter>
#>   <use href="#circles" filter="url(#colorMeMatrix)" transform="translate(0 70)" />
#>   <text x="70" y="120">
#>     Matrix
#>   </text>
#>   <filter id="colorMeSaturate">
#>     <feColorMatrix in="SourceGraphic" type="saturate" values="0.2" />
#>   </filter>
#>   <use href="#circles" filter="url(#colorMeSaturate)" transform="translate(0 140)" />
#>   <text x="70" y="190">
#>     Saturate
#>   </text>
#> </svg>


Reference Matrix Saturate

feTurbulence

The <feTurbulence> SVG filter primitive creates an image using the Perlin turbulence function. It allows the synthesis of artificial textures like clouds or marble. The resulting image will fill the entire filter primitive subregion.

Example from MDN docs.

#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
# Define filter with turbulence driving the displacmenet
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
my_filter <- stag$filter(
  id = "displacementFilter",
  stag$feTurbulence(
    type          = "turbulence",
    baseFrequency = 0.05,
    numOctaves    = 2,
    result        = "turbulence"),
  stag$feDisplacementMap(
    in_   = "SourceGraphic",
    in2   = "turbulence",
    scale = 50,
    xChannelSelector = 'R',
    yChannelSelector = 'G')
)


#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
# Apply this displacement filter to a circle
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
doc <- svg_doc(
  width = 200, height = 200,
  my_filter,
  stag$circle(cx=100, cy=100, r=80, filter = my_filter)
)

Show SVG text (click to open)

#> <?xml version="1.0" encoding="UTF-8"?>
#> <svg width="200" height="200" viewBox="0 0 200 200" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
#>   <filter id="displacementFilter">
#>     <feTurbulence type="turbulence" baseFrequency="0.05" numOctaves="2" result="turbulence" />
#>     <feDisplacementMap in="SourceGraphic" in2="turbulence" scale="50" xChannelSelector="R" yChannelSelector="G" />
#>   </filter>
#>   <circle cx="100" cy="100" r="80" filter="url(#displacementFilter)" />
#> </svg>


feTurbulence - simple example

doc <- svg_doc(
  width = 250, height = 250,
  stag$filter(
    id = "noise",
    stag$feTurbulence(baseFrequency = 0.05)
  ),

  stag$rect(
    x      = 0,
    y      = 0,
    height = 250,
    width  = 250,
    filter = "url(#noise)",
    fill   = 'none')
)

Show SVG text (click to open)

#> <?xml version="1.0" encoding="UTF-8"?>
#> <svg width="250" height="250" viewBox="0 0 250 250" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
#>   <filter id="noise">
#>     <feTurbulence baseFrequency="0.05" />
#>   </filter>
#>   <rect x="0" y="0" width="250" height="250" filter="url(#noise)" fill="none" />
#> </svg>