Compile Your SVG Sprites With Gulp

back

No matter what kind of app you’re building, chances are you’re going to need icons at some point. As of this writing, the best, most accessible method of working with icon sprites is to use SVG graphics. This can lead to a lot of tedious copy and paste of SVG code into a symbol library for optimization. Or… we could have gulp take care of that for us!

As you may remember from my previous post about using an SVG icon store with Vue.js, symbol libraries are convenient to use, and eliminate the need to download an extra sprite sheet old school style. When writing the article for the Vue.js icon store, there ended up being a fair amount of copy/paste work and some custom editing of SVG files in a text editor. That’s all well and good, but we can do better.

Project Setup

Let’s make a new project named gulp-sprites with the following structure:

gulp-sprites-demo
├── index.html
├── gulpfile.js
└── sprites

Nothing special or crazy, just a gulpfile, index.html for setting up our example, and a sprites folder to dump our sprites in. Speaking of sprites, head on over to SimpleIcons.org and download a few icons to the sprites folder. Based on what I downloaded, my project now looks like this:

gulp-sprites-demo
├── index.html
├── gulpfile.js
└─┬ sprites
  ├── facebook.svg
  ├── instagram.svg
  ├── pinterest.svg
  └── twitter.svg

We’ll also need to install some dependencies and then set up our index.html file with a basic boilerplate.

cd /path/to/project/root
npm init
npm install --save gulp gulp-svgstore gulp-inject
<!-- index.html -->
<!DOCTYPE html>
<html>
  <head>
    <title>Compile Your SVG Sprites With Gulp</title>
  </head>
  <body>
    <!-- inject:svg --><!-- endinject -->
  </body>
</html>

Compiling the Store

Now it’s time to crack open that gulp file and compile those SVGs.

var gulp = require('gulp');
var svgstore = require('gulp-svgstore');
var inject = require('gulp-inject');

gulp.task('default', function () {
  var svgs = gulp.src('sprites/*.svg').pipe(svgstore({ inlineSvg: true }));

  function fileToString(filePath, file) {
    return file.contents.toString();
  }

  return gulp.src('./index.html')
    .pipe(inject(svgs, { transform: fileToString }))
    .pipe(gulp.dest('./'));
});

So in our default task we are using gulp-svgstore to grab all of our icons from the sprites directory and parse them as inline SVGs. We can then pipe those inline SVGs into our inject code in index.html. Pretty sweet, right?

Now that we’ve got all of our icons compiling into our markup, they’ll be available anywhere on the page through our auto-magic symbol library.

<!-- index.html -->
<body>
  <!-- ... -->
  <svg>
    <use xmlns:xlink="http://www.w3.org/1999/xlink" xlink:href="#facebook"></use>
  </svg>
  <svg>
    <use xmlns:xlink="http://www.w3.org/1999/xlink" xlink:href="#instagram"></use>
  </svg>
  <svg>
    <use xmlns:xlink="http://www.w3.org/1999/xlink" xlink:href="#pinterest"></use>
  </svg>
  <svg>
    <use xmlns:xlink="http://www.w3.org/1999/xlink" xlink:href="#twitter"></use>
  </svg>
</body>

The biggest caveat here has to do with naming. Be default, gulp-svgstore will use the name of the SVG file as the ID of each individual sprite, which is why we’ve used the #facebook ID to reference the graphic in the facebook.svg file. For the full list of features, and many more configuration examples, you can check out the gulp-svgstore Github page.

Want to DRY up your code a little more? Check out my article on converting an SVG store to a Vue.js component.

View the code on Github.