PostCSS plugin for creating SVG sprites.
Input
.img {
background: url('img.svg');
}
Output
.img {
background: url('sprite.svg') no-repeat 0 0;
background-size: 100% 104.50%; /* Bg size calculated to scale image proportionally */
}
npm install postcss-svg-mixer
const { writeFileSync } = require('fs');
const postcss = require('postcss');
const mixer = require('postcss-svg-mixer');
postcss()
.use(mixer())
.process('.img {background: url(img.svg)}')
.then(result => {
const msg = result.messages.find(m => m.kind === 'sprite');
writeFileSync(msg.filename, msg.content);
});
const mixer = require('postcss-svg-mixer');
module.exports = {
plugins: [
mixer()
]
}
- Find
background
andbackground-image
declarations which containsurl()
part. - Trying to resolve file referenced in first
url()
occurrence. If URL starts with tilde~
plugin will search it in node_modules (Node.jsrequire.resolve
mechanism is used). If file exists - read it's content and add to sprite, throw error otherwise. - Modify original background image declaration for properly positioning symbol on sprite canvas, eg:
.img { background: url('img.svg') } /* becomes */ .img { background: url('sprite.svg') no-repeat 0 96.15%; background-size: 101.83% 217.01%; }
- Generate sprite and add a message to
result.messages
with sprite content, filename and instance with following format:{ type: string = 'asset', kind: string = 'sprite', plugin: string = 'postcss-svg-mixer', file: string, sprite: Sprite | StackSprite, filename: string, content: string }
Type:
string
Default:'classic'
Possible values:
classic
(default). Images placed on canvas one after the other. Works perfect in all browsers including IE10+. Generate additional styles for background image positioning.stack
. SVG stacking technique - images placed one below the other and hidden by default. Target image becomes visible via CSS:target
selector when referencing sprite, eg.sprite.svg#twitter
. Doesn't work in Safari prior to 11.1 macOS and 11.3 iOS. Don't generate additional styles for background image positioning.
Type:
string
Default:'sprite.svg'
Sprite filename which used in generated styles and in result message.
Type:
string | RegExp | Array<string | RegExp>
Default:/\.svg(\?.*)?$/
(any SVG file with optional query param, egimg.svg?qwe=123
)
Filter which images should be added to sprite. Could be a string (glob pattern),
RegExp or array of them. Rules are tested against absolute image path. If URL
starts with tilde ~
plugin will search image in node_modules (Node.js
require.resolve
mechanism is used).
Type:
string
Default:null
By default plugin transforms current rule, but is is possible to create separate rule with sprite styles by specifying a valid CSS selector. Note that original background image declaration will be moved to new rule. Example:
mixer({ selector: '::after' })
/* Input */
.img {
background: url('img.svg');
}
/* Output */
.img {}
.img::after {
background: url('sprite.svg') no-repeat 0 0;
background-size: 100% 104.50%;
}
Type:
Sprite
|StackSprite
Default:null
This plugin can be used as style generator for existing svg-mixer sprite instance.
const { writeFileSync } = require('fs');
const createSprite = require('svg-mixer');
const postcssMixer = require('postcss-svg-mixer');
// Create sprite programmatically
createSprite('img/*.svg').then(result => {
writeFileSync(result.filename, result.content);
/**
* Generate CSS code like
* .img1 {background: url(img/img1.svg);}
* .img2 {background: url(img/img2.svg);}
*/
const cssInput = result.sprite.symbols
.map(s => `.${s.id} {background: url(${s.image.path});}`)
.join('\n');
postcssMixer
.process(cssInput, { from: __filename }, { userSprite: result.sprite })
.then(({ css }) => writeFileSync('output.css', css));
});