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:
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("http://www.wallpapers-web.com/data/out/191/5484624-sunset-wallpapers.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("http://www.wallpapers-web.com/data/out/191/5484624-sunset-wallpapers.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?
There are two ways to do this...
.acrylic
Complex implementation but runs on most modern browsers.backdrop-filter
on.acrylic
Very simple implementation but lacks browser support.1. Blurred bg on
.acrylic
We need to duplicate bg on
.acrylic
guys too, because just bringing down opacity will show the content behind them not in them, which AFAIK is not covered by blur filter...Smart way to calc positions would be to set
background-attachment: fixed
for both parent element (body
) and.acrylic
guys, this will allow you to have multiple.acrylic
guys as well ;)Since we use same background for parent and children, we can club them together ;)
Here is a working snippet ;) Turned opacity on
.acrylic:after
a bit down so background is a bit more visible ;)2. (Less supported)
backdrop-filter
on.acrylic
As far as I know there is no smarter way in this case to achieve that, since the
background image
and thefilter
have to be set to the same div to create this desired effect.As far as I understood, the issue is that with every additional div
.acrylic
the background part of the image gets duplicated.Since you're using the same background image you could fix this by simply adding
background-attachment: fixed
to the div.acrylic::before
.This answer solves the problem and I've enhanced it aswell.
I think the only problem you had was background alignment applied to the ::before and the body were different. So by combining the CSS used into one, I've shortcutted this to make updating and editing it easier.
I used background:rgba() instead of #hex values to give more flexibility over the opacity controls.
I've added notes throughout the CSS to explain the changes I've made to make them more apparent.
NOTE: Removing the opacity from the ::after class causes transparency to increase and increases the visibility of the noise texture. I believe it is being caused by the noise texture itself being semi transparent, but it is an odd behaviour, if anyone else knows and has a better explanation I'd be interested to know. I've added classes .nested and .parent. as it may be preferable to have control over these things separately. I've removed .parent in the second example and both in the third.
Update: Noticed the exclusion blend filter mentioned in spec wasn't included, so I've added that in too. Provided sources below the snippet.
Hope this helped. :)
CSS & HTML Snippet with 3 examples:
Sources and extra notes:
You can find a source for front glass effect on CSS Tricks here.
They also say that if you want to offer even more browser support you can add in SVG filter as a fall back, although I haven't added it to my own snippet they say this:
And provided a code example
SVG:
CSS
For more about the CSS filter blends, go here https://css-tricks.com/basics-css-blend-modes