cssgaussianblurfluent-designacrylic-material

CSS-only Acrylic Material from Fluent Design System


With the advent of Microsoft's Fluent Design System and the propagation of the new Acrylic Material around the Windows ecosystem, I thought it would be great to use it in some Web layouts.

Acoording to the spec, the composition of an acrylic layer is:

Image

So I went to try a CSS-only approach inspired by the layers in that picture, this way:

body {
  margin: 0;
  font: 1em/1.4 Sans-serif;
  background: url("https://cdn.pixabay.com/photo/2017/03/27/16/50/beach-2179624_1280.jpg") center center;
  background-size: 100vw auto;
}

main {
  display: flex;
  align-items: center;
  justify-content: center;
  height: 100vh;
}

.acrylic {
  padding: 4em 6em;
  position: relative;
  overflow: hidden;
}

.acrylic::before {
  background: url("https://cdn.pixabay.com/photo/2017/03/27/16/50/beach-2179624_1280.jpg") center center;
  background-size: 100vw auto;
  filter: blur(10px);
  content: "";
  position: absolute;
  left: -10px;
  top: -10px;
  width: calc(100% + 20px);
  height: calc(100% + 20px);
  z-index: -1;
}

.acrylic::after {
  content: "";
  position: absolute;
  left: 0;
  top: 0;
  right: 0;
  bottom: 0;
  z-index: -1;
  opacity: 0.65;
  border: 1px solid #fff;
  background: #fff;
  background-image: url();
}

.shadow {
  border-radius: 1px;
  box-shadow: 0 10px 30px rgba(0, 0, 0, 0.1), 0 1px 8px rgba(0, 0, 0, 0.2);
}
<main>
  <div class="acrylic shadow">
    Acrylic material!
  </div>
</main>

The result is really close to the spec and is also responsive, but has a big problem: just stack another .acrylic div and the background trick doesn't work anymore.

The question is: is there some smarter way to gaussian blur without duplicating the body background for each children? Or maybe some smarter way to dinamically calculate its position?


2022 Update

When I originally posted this question, the backdrop-filter CSS feature was an experimental thing not enabled even in Chrome, so I was asking for an alternative solution, if possible, at the time.

Time passes... Finally it's enabled by default on all browsers over 0.5% users. So it's really simplier to solve nowadays than it was before.


Solution

  • 2022 Update

    TL;DR: Use backdrop-filter for the blur effect and that's it.

    When I originally posted the question, the backdrop-filter CSS feature was an experimental thing not enabled even in Chrome, so I was asking for an alternative solution, if possible, at the time.

    Since then, the backdrop-filter feature shipped in Chrome 76 at July 29ᵗʰ 2019, already turning it into a viable option for progressive enhancement use cases. Finally, after years of recurring postpones, Firefox released it enabled by default in 103 version, at July 26ᵗʰ 2022.

    Now with a theoretical cross-browser support on about 95%, and a practical 100% support considering browsers with more than 0.5% of users, it's clearly the option to go:

    /* Now all the acrylic layer is just only one class! */
    .acrylic {
      /* Parent background + Gaussian blur */
      backdrop-filter: blur(10px);
      -webkit-backdrop-filter: blur(10px); /* Safari */
    
      /* Exclusion blend */
      background-blend-mode: exclusion;
    
      /* Color/tint overlay + Opacity */
      background: rgba(255, 255, 255, .6);
    
      /* Tiled noise texture */
      background-image: url();
    }
    
    /* Other styles for sample purposes... */
    body {
      margin: 0;
      padding: 1.5em;
      font: 1em/1.4 Sans-serif;
    
      /* Now our background image is defined only in the body! */
      background: url("https://cdn.pixabay.com/photo/2017/03/27/16/50/beach-2179624_1280.jpg") center center;
      background-size: 100vw auto;
    }
    
    main {
      display: grid;
      gap: 1.5rem;
      justify-content: center;
      grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
      grid-auto-rows: minmax(120px, auto);
    }
    
    div {
      padding: 1.5em;
      border-radius: 1px;
      border: 1px solid rgba(255, 255, 255, .2);
      box-shadow: 0 10px 30px rgba(0, 0, 0, .1), 0 1px 8px rgba(0, 0, 0, .2);
      text-align: center;
      display: flex;
      align-items: center;
      justify-content: center;
    }
    <main>
      <div class="acrylic">
        Acrylic material!
      </div>
    
      <div class="acrylic">
        <div class="acrylic">
          Acrylic inside acrylic!
        </div>
      </div>
    
      <div class="acrylic">
        Acrylic material!
      </div>
    
      <div class="acrylic">
        Acrylic material!
      </div>
    </main>