Developed for nearly two years of small program, has been quietly in small program iteration. I haven’t played Vue for a long time. Recently, I used Vue in a new project. When I saw too many small ICONS in the Ui image, I immediately thought of using Sprite image

Finally, I recommend webpack-Spritesmith, which is a fusion node to generate CSS + Sprite layout tool, you can import CSS, using the label like iconfont can be called, more convenient

Making: github.com/mixtur/webp…

You can customize the style template and then customize the tags, whether stylus/SCSS/LESS/CSS

1. Don’t say anything, install ~

npm i spritesheet-templates --save-dev
Copy the code

2. Batch rename images first

Why is that? Neat and good-looking, like a parade ceremony, in fact, in order, is convenient for your CSS table to find better

3. Once you have a patchwork icon library, start configuring WebPack

The directory structure

# according to the instructions / | - SRC | | - the ICONS | | | - icon_01. PNG | | | - icon_02. PNG | | | - icon_03. PNG | |... | |-style.styl | ... |- webpack.config.jsCopy the code

General configuration

var path = require('path');
var SpritesmithPlugin = require('webpack-spritesmith');
module.exports = {
    // ...
    module: {
        rules: [{test: / \.styl$/, use: ['style-loader',
                'css-loader',
                'stylus-loader']}, {test: / \.png$/, use: ['file-loader?name=i/ [hash]. [ext]] ']}},resolve: {
        modules: ["node_modules","spritesmith-generated"]},plugins: [
        new SpritesmithPlugin({
            src: {
                cwd: path.resolve(__dirname, 'src/icons'), # where the icon is locatedglob: '*.png'# icon format},targetIt is recommended that the two files be placed togetherimage: path.resolve(__dirname, 'src/spritesmith-generated/sprite.png'), # generate a Sprite image at the specified locationcss: path.resolve(__dirname, 'src/spritesmith-generated/sprite.styl|css|stylus|scss|less] '),   # generate the specified position stylesheet that will correspond to all the background offset positions of the Sprite image},apiOptions: {# for the relative path of the stylesheet generated abovesprite.styl|css|stylus|scss|less]
                cssImageRef: ". /sprite.png"}})] //... };Copy the code

Custom style templates

The second configuration method, generic, is this custom style template

# Division accuracy problem
function divide (arg1, arg2) {
    console.log(arg1, arg2)
    var t1 = 0; var t2 = 0; var r1; var  r2
    try { t1 = arg1.toString().split('. ') [1].length} catch (e) {} // -- length after decimal pointtry { t2 = arg2.toString().split('. ') [1[.length} catch (e) {} r1 = Number(arg1.toString().replace())'. '.' 'R2 = Number(arg2.tostring ().replace())'. '.' ') // -- Remove the decimal point to integerreturn (r1 / r2) * Math.pow(10, t2-t1) // -- The integer is divided by times10The length of the decimal point}# function custom stylesheet methods
 # emsp; If you have a
 function templateFunction(data) {
    var spritesheet = data.spritesheet # Sprite map object
    var unit = 'px' # unit
    var w = spritesheet.width # Sprite map width
    var h  = spritesheet.height # Sprite height
    var shared = `.ico { 
          background-image: url(${spritesheet.image});
          background-size: ${w}${unit} ${h}${unit} 
    }`
    
    # Loop Sprite and set the width and height of the corresponding icon style
    # data.sprites: Sprite small icon traversal
    var perSprite = data.sprites.map(function (sprite) {
        var offsetX = sprite.offset_x # the x-distance from the top left corner, which is negative, is calculated by default
        var offsetY = sprite.offset_y # The y distance from the top left corner, which is negative, is calculated by default
        # return and return a default style, because rem will have some deviation, so we need to set the padding:1px
        return `.ico-N {position:relative; top: 2${unit}; padding:1${unit}; width: ${sprite.width}${unit}; height: ${sprite.width}${unit}; background-position: X${unit} Y${unit}; display: inline-block; }` .replace('N', sprite.name)
            .replace('X', offsetX)
            .replace('Y', offsetY)
    }).join('\n')

    return shared + '\n' + perSprite
}



module.exports = {
        ...
        plugins: [
            new SpritesmithPlugin({
                src: {
                    cwd: path.resolve(__dirname, 'src/icons'), # The position of the icon
                    glob: '*.png' The format of the icon
                },
                target: {
                    image: path.resolve(__dirname, 'src/spritesmith-generated/sprite.png'), # Generate Sprite image at specified location
                    css: [
                        # Introduce style templates
                        [path.resolve(__dirname, 'src/spritesmith-generated/sprite.[styl|css|stylus|scss|less]'), {
                            format: 'function_based_template' The style template here is specified by customTemplates
                        }]
                    ]
                },
                apiOptions: {
                    cssImageRef: './sprite.png'
                },
                customTemplates: {
                    'function_based_template':  templateFunction, # specify the style template, which can be a function or handlebars background template
                },
                spritesmithOptions: {
                    # for the relative path to the generated stylesheet above Sprite. [styl | | CSS stylus | SCSS | less]
                    algorithm: 'top-down'.# Arrange Sprite images from top to bottom
                    padding: 20.# Sprite spacing}}})]Copy the code

4. Use it in labels

  • Open your icon to see the desired icon name
  • Copy the name of the icon you need, open just generated Sprite. [styl | | CSS stylus | SCSS | less]
  • Introducing the Sprite. [styl | | CSS stylus | SCSS | less] to the SRC/main js, home directory Such as:
  • Used in. Vue-template
    <div class="ico ico-icon_14"></div>
    <div class="ico ico-icon_13"></div>
    <div class="ico ico-icon_16"></div>
    <div class="ico ico-icon_17"></div>
    <div class="ico ico-icon_18"></div>
    <div class="ico ico-icon_19"></div>
    <div class="ico ico-icon_20"></div>
    <div class="ico ico-icon_21"></div>
Copy the code

This is equivalent to importing the template in your custom stylesheet

5. It is very convenient to use successfully

In the future, NPM run dev will be automatically generated if it is only needed to start. It is suggested that if the map is added later, it can be divided out, or it can be added later as needed for better maintenance

6. Rem pit

I used 75 pixels for 750 designs, so convert to 10rem = 750px / 1rem = 75px

  • Pit 1: Unit to position conversion problem used in the projectpostcss-px2rem-excludeOr some other transformation REM tool

ps: All of these tools will cause rem to automatically convert, and the position of the Sprite image will not correspond to the one provided by the style sheet, so you need to set background-size for each Sprite image. In fact, you can set background-size for each Sprite image. For example, if the Sprite image is 300px * 300px, then background-size is 300px and 300px. After setting the image, you will not encounter the problem of mismatch

  • Pit 2: Android – wechat QQ browser inside the corner deviation 1px

A good match for half a day, I met this pit, solved about 6 to 7 hours, after this problem is mainly due to rem conversion, calculated value deviation, caused by the external browser open all no problem, AnZhuoWei letter inside the QQ browser appears deviation problem, also can appear in other mobile phones and I tried to use % instead of, but failed, I have tried to calculate the precision problem with precise function. There will still be deviations, and generally we don’t know which link went wrong. Well, I don’t have to worry about it.

One is to add padding: 1px to the configuration stylesheet to make up for some of the accuracy in calculating the js presence when running back

. Second is the configuration of spritesmithOptions padding spacing a little bigger, Sprite figure direction change off the top – down, solves the rem within WeChat QQ browser

spritesmithOptions: {
    # for the relative path to the generated stylesheet above Sprite. [styl | | CSS stylus | SCSS | less]
    algorithm: 'top-down'.# Arrange Sprite images from top to bottom
    padding: 20, # Sprite spacing
}
Copy the code

Conclusion: It is impossible to think…