Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

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?


Update

Since the backdrop-filter CSS feature has finally shipped in Chrome 76 in July 29ᵗʰ, we have a really simplified approach to solve it, turning the original question a lot easier than it was before.

like image 587
Erick Petrucelli Avatar asked Jun 13 '17 13:06

Erick Petrucelli


People also ask

Is there a post CSS-only acrylic material from Fluent Design System?

I found this Post CSS-only Acrylic Material from Fluent Design System which is great but it has a big problem. When I try to use more then one background image it doesn´t work anymore, because the following piece of code is needed:

Is it possible to use acrylic material in web layouts?

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:

What does fluent do at Microsoft Build 2018?

For those of you who are coming away from Microsoft Build 2018 inspired to breathe some Fluent design into your experiences, here’s what the Fluent team does to create Acrylic Material on their transient surfaces on the web:

Is CSS “the Dark Arts”?

He went on to compare it to the more traditional forms of computer science ( math) and visual design ( art ). “But people with CSS skills, man…” he sighed, “it’s the dark arts.” This has been especially true for those of us who make web sites work across browsers and operating systems.


2 Answers

There are two ways to do this...

  1. Blurred bg on .acrylic Complex implementation but runs on most modern browsers.
  2. 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 ;)

body, .acrylic::before {   background: url("IMG_URL_HERE") center/cover;   background-attachment: fixed; } 

Here is a working snippet ;) Turned opacity on .acrylic:after a bit down so background is a bit more visible ;)

body {    margin: 0;    font: 1em/1.4 Sans-serif;  }    body, .acrylic::before {    background: url("https://images.unsplash.com/photo-1452723312111-3a7d0db0e024?w=700") center/cover;    background-attachment: fixed;  }    main {    display: flex;    align-items: center;    justify-content: center;    height: 100vh;  }    .acrylic {    padding: 4em 6em;    position: relative;    overflow: hidden;    margin: 1em;  }    .acrylic::before {    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.35;    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>    <div class="acrylic shadow">      Acrylic material!    </div>  </main>    <main>    <div class="acrylic shadow">      Acrylic material!    </div>    <div class="acrylic shadow">      Acrylic material!    </div>  </main>    <main>    <div class="acrylic shadow">      Acrylic material!    </div>  </main>

2. (Less supported) backdrop-filter on .acrylic

main {    display: flex;    align-items: center;    justify-content: center;    height: 100vh;  }    .acrylic {    padding: 4em 6em;    position: relative;    background: rgba(0,0,0,0.5);    -webkit-backdrop-filter: blur(10px);    backdrop-filter: blur(10px);    margin: 7px;  }    body {    background: url("https://images.unsplash.com/photo-1452723312111-3a7d0db0e024?w=700") center/cover;    background-attachment: fixed;    margin: 0;    font: 1em/1.4 Sans-serif;    color: #fff;  }
<main>    <div class="acrylic shadow">      Acrylic material!    </div>    <div class="acrylic shadow">      Acrylic material!    </div>  </main>    <main>    <div class="acrylic shadow">      Acrylic material!    </div>    <div class="acrylic shadow">      Acrylic material!    </div>  </main>    <main>    <div class="acrylic shadow">      Acrylic material!    </div>  </main>
like image 153
shramee Avatar answered Oct 10 '22 22:10

shramee


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:

/* Adding the background source for   both elements together simplifies editing.   Changing the background color #333 to #fff   will produce much more intence effect on   the filter blend */    html,  .acrylic:before {    background: #333 url(https://source.unsplash.com/1600x900/?nature) 50% 100% fixed;    background-size: cover;  }    body {    margin: 0;    font: 1em/1.4 Sans-serif;  }      /*div spacings*/    div {    margin: 5px;  }    main {    display: flex;    align-items: center;    justify-content: center;    flex-flow: row wrap;    height: 100vh;  }    .acrylic {    padding: 4em 6em;    position: relative;    overflow: hidden;  }      /*Added browser compatibility for blurring added the exclusion blend filter as explained in the original document*/    .acrylic::before {    content: '';    position: absolute;    z-index: -1;    height: 100%;    top: 0;    right: 0;    left: 0;    filter: blur(8px);    -webkit-filter: blur(8px);    -moz-filter: blur(8px);    -o-filter: blur(8px);    -ms-filter: blur(8px);    background-blend-mode: exclusion;  }      /*made color rgba removed opacity property*/    .acrylic::after {    content: "";    position: absolute;    height: 100%;    left: 0;    top: 0;    right: 0;    bottom: 0;    z-index: -1;    border: 1px solid #fff;    background-image: url();  }      /*Individual control over   .nested and .parent opacity and color.*/    .parent::after {    background-color: rgba(230, 240, 255, 0.50);    opacity: 0.60;  }    .child::after {    background-color: rgba(230, 240, 255, 0.30);    opacity: 0.60;  }    .shadow {    box-shadow: 0 10px 30px rgba(0, 0, 0, 0.2), 0 1px 8px rgba(0, 0, 0, 0.4);  }
<body>    <main>      <div class="acrylic shadow parent">        <div class="acrylic child">          Acrylic material! <br />.parent on .child on        </div>      </div>      <div class="acrylic shadow">        <div class="acrylic child">          Acrylic material! <br />.parent off .child on        </div>      </div>      <div class="acrylic shadow">        <div class="acrylic">          Acrylic material! <br />.parent off .child off        </div>      </div>    </main>  </body>

Sources and extra notes:

You can find a source for front glass effect on CSS Tricks here.

The duplicate image technique requires maintaining a blurred image along with the original, which can become a pain if you need to reuse the effect for multiple images. For example, responsive designs may require swapping in different images at different screen sizes. Or, template layouts may drop in images dynamically (eg, a different header image for every blog post). For these cases, it would be nice to generate the effect using only the source image. After all, we're just blurring it.

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:

CSS Filters are somewhat new. That means they may be vendor prefixed, and that their browser support is not yet universal. However, filters have a longer history in SVG, and applying SVG filters to HTML content via CSS has wider browser support. You can easily add them as a fallback for when CSS filters are not supported. The above demo actually does just that.

To add an SVG filter, we include some inline SVG in our HTML markup, and reference the filter with a url(). Pro tip: An alternative is to encode the SVG filter and reference as a data url, but that format is a bit more difficult to read in an article.

And provided a code example

SVG:

<svg xmlns="http://www.w3.org/2000/svg" version="1.1">   <defs>     <filter id="blur">       <feGaussianBlur stdDeviation="5" />     </filter>   </defs> </svg> 

CSS

.glass::before {     background-image: url('pelican.jpg');     /* Fallback to SVG filters */     filter: url('#blur');     filter: blur(5px); } 

For more about the CSS filter blends, go here https://css-tricks.com/basics-css-blend-modes

like image 29
GibsonFX Avatar answered Oct 10 '22 23:10

GibsonFX