CSSのinterpolate-sizeプロパティを使ったアニメーションの許可
昨今はCSSのみで様々なアニメーションが実装できるようになりましたが、コンテンツ量が変わると多少の調整が必要になる場合もあります。
あらかじめ決まっているコンテンツの要素の幅をアニメーションさせる場合は、transitionプロパティやwidthプロパティの組み合わせで、widthプロパティには決まった数値を指定することでアニメーションが実装できます。
しかし、コンテンツ幅が変化する可能性のある要素をアニメーションさせる場合は、CSSのみでは実装するのが難しいです。
コンテンツ幅が変化する可能性のある要素には、JavaScriptを使って要素の幅を取得してアニメーションさせるなど、少し手の込んだプログラムが必要になります。
ただ、その問題もCSSのinterpolate-sizeプロパティが解決してくれるでしょう。
CSSで「interpolate-size: allow-keywords;」を指定することで、widthプロパティのauto、min-content、max-contentなどの値をCSSアニメーションに適用することができます。
つまり、コンテンツ量に応じた幅に対してアニメーションが実装できるようになります。
「interpolate-size: allow-keywords;」は親要素以上で指定する必要がありますが、様々な要素で必要になることも考えると、ルート要素(:root)に指定するのがベストでしょう。
CSS
:root {
interpolate-size: allow-keywords;
}
気になるWebブラウザのサポート状況ですが、
2024年10月現在時点では、Google Chrome 129以降のみサポートしています。
Can I use (interpolate-size)
https://caniuse.com/?search=interpolate-size
Google Chromeがサポートされれば、ChromiumベースのMicrosoft Edgeなどもサポートされるので、あとはSafariやFirefoxのサポートを待つのみとなります。
interpolate-sizeプロパティを使ったアニメーションの実装
ここからは、interpolate-sizeプロパティを使ったアニメーションの実装を、サンプルを交えて確認していきます。
サンプルとして、お問い合わせ用のアイコンボタンの動作で見ていきます。
HTMLは以下のような構造を構築します。
HTML
<nav class="contact-nav">
<ul>
<li>
<a href="#">
<img decoding="async" src="" alt="mail">
<span class="nav-text">Mail</span>
</a>
</li>
<li>
<a href="https://bsky.app/profile/akirao.bsky.social" target="_blank" rel="noopener noreferrer">
<img decoding="async" src="" alt="SNS">
<div class="nav-text">Bluesky</div>
</a>
</li>
</ul>
</nav>
メールでのお問い合わせは、コンタクトフォームを設置しているページなどにリンクさせるといいでしょう。
チャットでのお問い合わせは、ご利用のチャットツールやSNSのDMで手軽にやりとりができるものに。ここでは誰でも簡単に無料でアカウントが作成できるBlueskyをリンクさせてみました。
続いてはCSS。
ルート要素(:root)に「interpolate-size: allow-keywords;」を指定しておきます。
他、お問い合わせのナビゲーションのスタイルを指定します。
以下、CSSのサンプルコードになります。
CSS
:root {
interpolate-size: allow-keywords;
}
.contact-nav {
ul {
display: flex;
flex-direction: column;
gap: 0.5rem;
list-style: none;
margin: 0;
padding: 1rem;
a {
display: grid;
grid-template-columns: 1.5rem auto;
gap: 1rem;
padding: 0.5rem;
align-items: center;
background: #ffd1c6;
border-radius: 0.5rem;
text-decoration: none;
font-size: 1rem;
color: #333;
width: 2.5rem;
overflow-x: clip;
white-space: nowrap;
transition: width 0.3s ease;
&:hover,
&:focus-visible {
width: max-content;
background: #ffab98;
}
}
}
img {
width: 1.5rem;
height: auto;
}
.nav-text {
padding-right: 0.5rem;
font-weight: bold;
}
}
今回のサンプルのポイントは、interpolate-sizeプロパティの指定のほか、ul要素のスタイルの部分になります。
リストのa要素のスタイルを見てみましょう。ポイントとなる部分がわかりやすいように、1行空けて記述しています。
「display: grid;」とグリッドレイアウトで構築しています。
grid-template-columnsプロパティで、横方向のグリッド線間の幅を指定します。アイコン部分が1.5rem、残りのコンテンツはautoとします。このときテキスト部分は1.5remからはみ出る状態です。
そしてコンテンツを折りたたんでいる部分は、横幅と切り取りで調整します。
「width: 2.5rem;」でアイコン部分となる幅を調整し、「overflow-x: clip;」でコンテンツを切り取りテキスト部分を非表示とします。
「white-space: nowrap;」は、もし改行したコンテンツを含めたときに、高さが変わった影響でデザインが崩れるので、もしものために指定しておきます。
アニメーションは「transition: width 0.3s ease;」で、横幅を0.3秒かけて変化させています。イージングはお好みの変化を指定してください。
hover擬似クラスまたはキーボード操作時にフォーカスした場合のfocus-visible擬似クラスに対して、横幅を「width: max-content;」とアイコンとテキストを含んだコンテンツの最大幅に指定します。
親要素以上に「interpolate-size: allow-keywords;」が指定してあることで、max-contentでもアニメーションに適用させることができます。
ちなみに、メニューを画面に固定する場合は、「position: fixed;」などで固定しましょう。
以下のCSSのスタイルと、確認のためにHTML要素も追加しておきます。
(サンプル:画面左中央に固定)
HTML(追加)
<section id="info">
<div class="section-conts">
</div>
</section>
<section id="about">
<div class="section-conts">
</div>
</section>
CSS(追加)
.contact-nav {
position: fixed;
top: 50%;
left: 0;
transform: translateY(-50%);
}
section {
min-height: 100vh;
&:nth-of-type(even) {
background-color: #eee;
}
}
infoとaboutのセクションを追加したので、スクロールしても画面左中央にメニューが固定されているのが確認できます。
最後に
今後、CSSのinterpolate-sizeプロパティが多くのWebブラウザでサポートされれば、開閉式のコンテンツやコンパクトなメニューボタンなどを、JavaScriptを使わずCSSのみで手軽に作成することができるでしょう。
ぜひ参考にしてください。