web-dev-qa-db-fra.com

Matériau acrylique uniquement CSS de Fluent Design System

Avec l'avènement de Fluent Design System de Microsoft et la propagation du nouveau matériau acrylique dans l'écosystème Windows, j'ai pensé qu'il serait formidable de l'utiliser dans certaines mises en page Web.

Selon la spécification , la composition d'une couche acrylique est:

Image

Je suis donc allé essayer une approche CSS uniquement inspirée des couches de cette image, de cette façon:

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>

Le résultat est vraiment proche de la spécification et est également réactif, mais a un gros problème: empilez simplement un autre .acrylic div et l'astuce d'arrière-plan ne fonctionne plus.

La question est: existe-t-il un moyen plus intelligent de flou gaussien sans dupliquer l'arrière-plan du corps pour chaque enfant? Ou peut-être une façon plus intelligente de calculer sa position de façon dinamique?


Mise à jour

Depuis le backdrop-filter La fonctionnalité CSS a enfin expédiée en Chrome 76 en juillet 29ᵗʰ, nous avons une approche vraiment simplifiée pour le résoudre , en tournant l'original question beaucoup plus facile qu'avant.

25
Erick Petrucelli

Le backdrop-filter La fonctionnalité CSS a enfin livrée en Chrome 76 depuis le 29 juillet. Malheureusement, Firefox y travaille toujours, devrait être livré dans un Q3 Version 2019 . Mais au moins, c'est clairement la meilleure option en ce moment pour les cas d'utilisation d'amélioration progressive.

Donc, je crois que la meilleure réponse peut être aussi simple que:

/* 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>
0
Erick Petrucelli

Il y a deux façons de faire ça...

  1. Bg floue sur .acrylic Implémentation complexe mais s'exécute sur la plupart des navigateurs modernes.
  2. backdrop-filter sur .acrylic Implémentation très simple mais manque de prise en charge du navigateur.

1. bg floue sur .acrylic

Nous devons dupliquer bg sur .acrylic les gars aussi, parce que simplement baisser l'opacité montrera le contenu derrière eux pas dedans, ce qui n'est pas couvert par AFAIK par le filtre flou ...

Un moyen intelligent de calculer des positions serait de définir background-attachment: fixed pour l'élément parent (body) et .acrylic les gars, cela vous permettra d'avoir plusieurs .acrylic les gars aussi;)

Puisque nous utilisons le même arrière-plan pour les parents et les enfants, nous pouvons les associer ensemble;)

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

Voici un extrait de code;) Opacité activée sur .acrylic:after un peu vers le bas pour que l'arrière-plan soit un peu plus 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. (Moins pris en charge) backdrop-filter sur .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>
24
shramee

Cette réponse résout le problème et je l'ai également amélioré.

Je pense que le seul problème que vous aviez était l'alignement d'arrière-plan appliqué au :: avant et le corps était différent. Donc, en combinant le CSS utilisé en un seul, j'ai raccourci cela pour faciliter la mise à jour et l'édition.

J'ai utilisé background: rgba () au lieu des valeurs #hex pour donner plus de flexibilité sur les contrôles d'opacité.

J'ai ajouté des notes tout au long du CSS pour expliquer les modifications que j'ai apportées pour les rendre plus apparentes.

REMARQUE: la suppression de l'opacité de la classe :: after augmente la transparence et augmente la visibilité de la texture du bruit. Je crois que cela est dû au fait que la texture du bruit elle-même est semi-transparente, mais c'est un comportement étrange, si quelqu'un d'autre le sait et a une meilleure explication, je serais intéressé de le savoir. J'ai ajouté des classes .nested et .parent. car il peut être préférable de contrôler ces choses séparément. J'ai supprimé .parent dans le deuxième exemple et les deux dans le troisième.

Mise à jour: j'ai remarqué que le filtre de mélange d'exclusion mentionné dans la spécification n'était pas inclus, j'ai donc ajouté cela également. Sources fournies sous l'extrait.

J'espère que cela vous a aidé. :)

Extrait CSS et HTML avec 3 exemples:

/* 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 et notes supplémentaires:

Vous pouvez trouver une source d'effet de vitre avant sur Astuces CSS ici.

La technique d'image en double nécessite de conserver une image floue avec l'original, ce qui peut devenir pénible si vous devez réutiliser l'effet pour plusieurs images. Par exemple, les conceptions réactives peuvent nécessiter l'échange de différentes images à différentes tailles d'écran. Ou, les mises en page de modèles peuvent déposer des images de manière dynamique (par exemple, une image d'en-tête différente pour chaque article de blog). Pour ces cas, il serait intéressant de générer l'effet en utilisant uniquement l'image source. Après tout, nous ne faisons que le brouiller.

Ils disent également que si vous souhaitez offrir encore plus de prise en charge du navigateur, vous pouvez ajouter un filtre SVG comme solution de rechange, bien que je ne l'aie pas ajouté à mon propre extrait, ils disent ceci:

Les filtres CSS sont quelque peu nouveaux. Cela signifie qu'ils peuvent être préfixés par le fournisseur et que la prise en charge de leur navigateur n'est pas encore universelle. Cependant, les filtres ont un historique plus long en SVG, et l'application de filtres SVG au contenu HTML via CSS a une prise en charge plus large du navigateur. Vous pouvez facilement les ajouter comme solution de rechange lorsque les filtres CSS ne sont pas pris en charge. La démo ci-dessus fait exactement cela.

Pour ajouter un filtre SVG, nous incluons du SVG en ligne dans notre balisage HTML et référençons le filtre avec une url (). Conseil de pro: Une alternative consiste à coder le filtre SVG et la référence en tant qu'url de données, mais ce format est un peu plus difficile à lire dans un article.

Et fourni un exemple de code

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

Pour en savoir plus sur les mélanges de filtres CSS, allez ici https://css-tricks.com/basics-css-blend-modes

12
GibsonFX

Existe-t-il un moyen plus intelligent de flou gaussien sans dupliquer le fond du corps pour chaque enfant?

Autant que je sache, il n'y a pas de moyen plus intelligent dans ce cas pour y parvenir, puisque le background image et filter doivent être réglés sur le même div pour créer cet effet souhaité.

Ou peut-être une façon plus intelligente de calculer dynamiquement sa position?

D'après ce que j'ai compris, le problème est qu'avec chaque div supplémentaire .acrylic la partie d'arrière-plan de l'image est dupliquée.

Puisque vous utilisez la même image d'arrière-plan, vous pouvez résoudre ce problème en ajoutant simplement background-attachment: fixed au div .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>
0
SvenL