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でもアニメーションに適用させることができます。

interpolate-sizeプロパティを使ったアニメーションの実装

ちなみに、メニューを画面に固定する場合は、「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のみで手軽に作成することができるでしょう。

ぜひ参考にしてください。