CSS
Compass

Compassを使ったSpriting最適化 [Retina+SVG編]

More than 1 year has passed since last update.

The end of sprites; the rise of SVG | Optical Cortex
というポストを見て、Compassにインラインデータを埋め込む機能があることを知りました。

これを使えば、これまで温めてきたCompassのSpritingに、SVGのサポートも加えることが出来そうです。

ディレクトリ構成

Compassのconfigで指定した画像フォルダ (ここでは sprites ) に、

sprites/icons_1x
├ icon-facebook.png
├ icon-twitter.png
└ icon-github.png

sprites/icons_2x
├ icon-facebook.png
├ icon-twitter.png
└ icon-github.png

sprites の隣に svg を作って、

svg/icons_svg
├ icon-facebook.svg
├ icon-twitter.svg
└ icon-github.svg

というように配置します。

mixin

前回、Compassを使ったSpriting最適化 [Retina編] で書いたmixinをベースに、SVGのサポートを加えます。

@import "compass/utilities/sprites";

// Mixin
$icons_1x-sprites: sprite-map( 'icons_1x/*.png', $spacing: 2px );
$icons_2x-sprites: sprite-map( 'icons_2x/*.png', $spacing: 4px );

@mixin get-icons-sprite( $sprite, $retina: false, $svg: false ) {
  $map-url: sprite-url( $icons_1x-sprites );
  $sprite-position: sprite-position( $icons_1x-sprites, $sprite );
  background: $map-url $sprite-position no-repeat;

  $sprite-image: sprite-file( $icons_1x-sprites, $sprite );
  width: image-width( $sprite-image );
  height: image-height( $sprite-image );

  @if $retina == true {
    @include media-retina {
      background-image: sprite-url( $icons_2x-sprites );
      background-size: image-width( sprite-path( $icons_1x-sprites ) ) auto;
    }
  }

  // SVGのサポート
  @if $svg == true {
    .svg & {
      background-image: inline-image( '../svg/icons_svg/#{$sprite}.svg', 'image/svg+xml' );
      background-size: image-width( $sprite-image ) image-height( $sprite-image );
      background-position: 0 0;
    }
  }
}

使用例

$icon-names: (
              'icon-facebook'
              'icon-twitter'
              'icon-github'
              );

@each $icon-name in $icon-names {
  .icons-#{ $icon-name } {
    display: inline-block;
    @include get-icons-sprite( #{ $icon-name }, true, true );
  }
}
<i class="icons-icon-facebook"></i>
<i class="icons-icon-twitter"></i>
<i class="icons-icon-github"></i>

生成されるCSSはこのようになります。(アイコン一つ分掲載)
SVGのデータがインラインに埋め込まれます。

.icons-icon-facebook {
  display: inline-block;
  background: url('../../assets/images/icons_1x-s3241a4cef7.png') 0 0 no-repeat;
  width: 48px;
  height: 48px; }
  @media only screen and (min-device-pixel-ratio: 1.5), only screen and (-o-min-device-pixel-ratio: 3 / 2), only screen and (-webkit-min-device-pixel-ratio: 1.5), only screen and (min-device-pixel-ratio: 1.5) {
    .icons-icon-facebook {
      background-image: url('../../assets/images/icons_2x-sda4f458d0f.png');
      background-size: 48px auto; } }
  .svg .icons-icon-facebook {
    background-image: url('');
    background-size: 48px auto;
    background-position: 0 0; }

SVGデータの圧縮

GruntでCompassを実行する際にSVGの圧縮プロセスを手前に挟むと良いでしょう。

sindresorhus/grunt-svgmin

※ Gruntのワークフロー一式は近日アップ予定です。

その他の考察

こちらで、icon-fontとSVGの比較や、data-uriによる埋め込み、sprite画像の生成について論じられています。
(以下適当訳)

  • icon-fontになっているのが一番楽に管理でき、単色での着色も簡単だが、位置合わせは手間がかかる。
  • SVGは複数色の着色や位置合わせが簡単だけど、素材作成に手間がかかる。
  • パフォーマンス面ではdata-uriよりspriteを利用する方が優れている。
  • サイズの大きなSVG SpriteはiOSでのレンダリングにバグあり? (960x560で描画されない)
  • PNGの方がSVGよりもレンダリングが速い。
  • spriteの自動生成をGruntで行う場合、セットアップにはかなり手間がかかる。

まとめ

素材の書き出しフローの検討 の時点から考えていましたが、

SVGをdata-uriで埋め込み、PNGでfallbackする

というのが、現時点では最もシンプルなコーディング方法である、と判断しました。