Акриловый материал только для CSS от Fluent Design System

С появлением Microsoft Fluent Design System и распространением нового акрилового материала в экосистеме Windows, я подумал, что было бы здорово использовать его в некоторых веб-макетах.

Согласно спецификации, состав акрилового слоя:

Образ

Итак, я попробовал CSS-подход, вдохновленный слоями на этой картинке, вот так:

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>

Результат действительно близок к спецификации и также отзывчив, но имеет большую проблему: просто сложите другой .acrylic div и трюк с фоном больше не работают.

Вопрос в том, есть ли какой-нибудь более умный способ размытия по Гауссу без дублирования фона тела для каждого ребенка? Или, может быть, какой-то более умный способ динамически рассчитать свою позицию?

3 ответа

Решение

Есть два способа сделать это...

  1. Размыто BG на .acrylic Сложная реализация, но работает в большинстве современных браузеров.
  2. backdrop-filter на .acrylic Очень простая реализация, но не поддерживает браузер.

1. Размытые BG на .acrylic

Нам нужно дублировать BG на .acrylic ребята тоже, потому что простое снижение непрозрачности покажет содержимое позади них, а не в них, что AFAIK не покрывается фильтром размытия...

Умный способ рассчитать позиции будет установить background-attachment: fixed для обоих родительских элементов (body) а также .acrylic ребята, это позволит вам иметь несколько .acrylic ребята так же;)

Так как мы используем один и тот же фон для родителей и детей, мы можем объединить их вместе;)

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

Вот рабочий фрагмент;) Включил непрозрачность .acrylic:after немного вниз, так что фон немного более заметен;)

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. (Менее поддерживается) backdrop-filter на .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>

Этот ответ решает проблему, и я также улучшил ее.

Я думаю, что единственной проблемой, с которой вы столкнулись, было выравнивание фона, примененное к::before и телу. Поэтому, объединив используемый CSS в один, я сократил это, чтобы упростить его обновление и редактирование.

Я использовал background:rgba() вместо значений #hex, чтобы обеспечить большую гибкость по сравнению с элементами управления непрозрачностью.

Я добавил примечания по всему CSS, чтобы объяснить изменения, которые я сделал, чтобы сделать их более очевидными.

ПРИМЕЧАНИЕ. Удаление непрозрачности из класса::after приводит к увеличению прозрачности и увеличивает видимость текстуры шума. Я полагаю, что это вызвано тем, что сама текстура шума является полупрозрачной, но это странное поведение, если кто-то еще знает и имеет лучшее объяснение, которое мне было бы интересно узнать. Я добавил классы.nested и.parent. так как может быть предпочтительнее контролировать эти вещи отдельно. Я удалил.parent во втором примере и оба в третьем.

Обновление: заметил, что фильтр исключающей смеси, упомянутый в спецификации, не был включен, поэтому я тоже добавил это. Предоставленные источники ниже фрагмента.

Надеюсь, это помогло.:)

Фрагмент CSS & HTML с 3 примерами:

/* 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>

Источники и дополнительные заметки:

Вы можете найти источник эффекта переднего стекла для CSS Tricks здесь.

Техника дублированного изображения требует сохранения размытого изображения вместе с оригиналом, что может стать проблемой, если вам нужно повторно использовать эффект для нескольких изображений. Например, адаптивный дизайн может потребовать замены в разных изображениях на экранах разных размеров. Или макеты шаблонов могут вставлять изображения динамически (например, разные заголовки для каждого сообщения в блоге). Для этих случаев было бы неплохо создать эффект, используя только исходное изображение. В конце концов, мы просто размываем это.

Они также говорят, что если вы хотите предложить еще большую поддержку браузера, вы можете добавить фильтр SVG в качестве запасного варианта, хотя я не добавил его в свой собственный фрагмент, они говорят следующее:

Фильтры CSS несколько новы. Это означает, что они могут иметь префикс поставщика, и что их поддержка браузера еще не универсальна. Однако фильтры имеют более длинную историю в SVG, и применение фильтров SVG к содержимому HTML через CSS имеет более широкую поддержку браузера. Вы можете легко добавить их в качестве запасного варианта, когда фильтры CSS не поддерживаются. Приведенная выше демка фактически делает именно это.

Чтобы добавить фильтр SVG, мы включаем в разметку HTML несколько встроенных SVG и ссылаемся на фильтр с помощью url (). Совет для профессионалов: альтернативой является кодирование SVG-фильтра и ссылки в качестве URL-адреса данных, но этот формат немного сложнее для чтения в статье.

И предоставил пример кода

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);
}

Чтобы узнать больше о CSS-фильтрах, перейдите по ссылке https://css-tricks.com/basics-css-blend-modes/

backdrop-filter Функция CSS наконец-то появилась в Chrome 76 с 29 июля. К сожалению, Firefox все еще работает над ним, ожидается, что он будет выпущен в 3-м квартале 2019 года. Но, по крайней мере, сейчас это явно лучший вариант для прогрессивных вариантов использования.

Поэтому я считаю, что лучший ответ может быть таким простым:

/* Now all the acrylic layer is just only one class! */
.acrylic {
  /* Parent background + Gaussian blur */
  backdrop-filter: blur(10px);
  -webkit-backdrop-filter: blur(10px);

  /* Exclusion blend */
  background-blend-mode: exclusion;

  /* Color/tint overlay + Opacity */
  background: rgba(255, 255, 255, .6);

  /* Tiled noise texture */
  background-image: url();

  /* Some unrelated styling... */
  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;
}

/* Body and main styles... */
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);
}
<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>

Есть ли какой-нибудь более умный способ размытия по Гауссу без дублирования фона тела для каждого ребенка?

Насколько я знаю, в этом случае нет более разумного способа добиться этого, поскольку background image и filter должны быть установлены на тот же div, чтобы создать этот желаемый эффект.

Или, может быть, какой-то умный способ динамически рассчитать свою позицию?

Насколько я понял, проблема в том, что с каждым дополнительным .acrylic фоновая часть изображения дублируется.

Поскольку вы используете одно и то же фоновое изображение, вы можете исправить это, просто добавив background-attachment: fixed в див .acrylic::before,

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;
  background-attachment: fixed;
  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>
  <div class="acrylic shadow">Acrylic material!</div>
  <div class="acrylic shadow">Acrylic material!</div>
</main>

Другие вопросы по тегам