Fluent Design SystemのCSSのみのアクリル素材
Microsoftの Fluent Design System の出現と、Windowsエコシステム全体での新しいアクリルマテリアルの普及により、一部のWebレイアウトでそれを使用するのは素晴らしいと思いました。
仕様 に従って、アクリル層の組成は次のとおりです:
そこで、この写真のレイヤーに触発されたCSSのみのアプローチを次のように試しました。
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>
結果は本当に仕様に近く、応答性にも優れていますが、大きな問題があります:ただ別の.acrylic
divおよびバックグラウンドトリックはもう機能しません。
問題は次のとおりです:各子供の体の背景を複製せずにガウスぼかしにスマートな方法はありますか?それとも、その位置を動的に計算するためのもっと賢い方法でしょうか?
更新
backdrop-filter
CSS機能がようやく Chrome 76 に出荷されました。7月29日、私たちは それを解決するための非常に単純化されたアプローチ 、元の質問を以前よりもずっと簡単に変えることができます。
backdrop-filter
CSS機能がついに shiped in Chrome 76 7月29日以来。残念ながら、Firefoxはまだ作業中です Q3 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>
これを行うには2つの方法があります...
.acrylic
複雑な実装ではbgがぼやけますが、ほとんどの最新ブラウザーで実行されます。backdrop-filter
on.acrylic
非常に単純な実装ですが、ブラウザのサポートがありません。
1. .acrylic
上のぼやけたbg
.acrylic
guysにもbgを複製する必要があります。不透明度を下げると、背後のコンテンツが表示されます。
位置を計算するスマートな方法は、親要素(body
)とbackground-attachment: fixed
guysの両方に.acrylic
を設定することです。これにより、複数の.acrylic
guysも持つことができます。 )
親と子に同じ背景を使用しているので、一緒にクラブできます;)
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
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>
この答えは問題を解決し、私もそれを強化しました。
あなたが持っていた唯一の問題は、:: beforeに適用されたバックグラウンドアライメントとボディが異なっていたと思います。そのため、使用するCSSを1つにまとめることで、これをショートカットして、更新と編集を簡単にしました。
不透明度コントロールの柔軟性を高めるために、#hex値の代わりにbackground:rgba()を使用しました。
CSS全体にメモを追加して、それらをより明確にするために行った変更を説明しました。
注::: afterクラスから不透明度を削除すると、透明度が増加し、ノイズテクスチャの可視性が増加します。ノイズテクスチャ自体が半透明であることが原因であると思いますが、他の誰かが私が知りたいと思うより良い説明を持っている場合、それは奇妙な振る舞いです。クラス.nestedおよび.parentを追加しました。これらのものを個別に制御することが望ましい場合があるためです。 2番目の例では.parentを削除し、3番目の例では両方を削除しました。
更新:仕様に記載されている除外ブレンドフィルターが含まれていないことに気づいたので、私もそれを追加しました。スニペットの下にソースを提供しました。
これが役に立てば幸いです。 :)
3つの例のCSSとHTMLスニペット:
/* 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トリックのフロントガラス効果のソースは、こちらで確認できます。
複製画像の手法では、元の画像とともにぼやけた画像を維持する必要があり、複数の画像に効果を再利用する必要がある場合は苦痛になります。たとえば、レスポンシブデザインでは、さまざまな画面サイズでさまざまな画像を交換する必要があります。または、テンプレートレイアウトは動的に画像をドロップする場合があります(たとえば、ブログ投稿ごとに異なるヘッダー画像)。これらの場合、ソース画像のみを使用して効果を生成するとよいでしょう。結局のところ、私たちはそれをぼかすだけです。
また、さらに多くのブラウザサポートを提供したい場合は、フォールバックとしてSVGフィルターを追加できると言いますが、自分のスニペットには追加していませんが、こう言います:
CSSフィルターはやや新しいものです。つまり、ベンダー接頭辞が付いている可能性があり、ブラウザのサポートはまだ一般的ではありません。ただし、SVGではフィルターの履歴が長くなり、CSSを介してHTMLコンテンツにSVGフィルターを適用すると、ブラウザーのサポートが広くなります。 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
各子供の身体の背景を複製せずにガウスぼかしを行うよりスマートな方法はありますか?
私が知る限り、この場合、background image
とfilter
を同じdivに設定してこの効果を作成する必要があるため、これを実現する賢い方法はありません。
それとも、その位置を動的に計算するより賢い方法でしょうか?
私が理解した限り、問題はdiv .acrylic
を追加するたびに画像の背景部分が複製されることです。
同じ背景画像を使用しているため、background-attachment: fixed
を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>