webpack之loader

1. 常见的loader

1.1 file-loader(处理静态资源模块)

原理是把打包过程中识别出的资源模块,移动到输出目录,并且返回新的地址名称。

使用场景:当我们需要的模块,仅仅是从源代码移动到打包目录,则可以使用file-loader来处理,比如txt, svg, csv, excel和图片资源等

安装
npm i file-loader -D
webpack配置
module:{
    rules:[
        {
            test:/\.(png|jpe?g|gif)$/,
            //use使⽤⼀个loader可以⽤对象,字符串,两个loader需要⽤数组
            use:{
                loader:' file-loader',
                // options额外的配置,⽐如资源名称
                options:{
                    name:'[name]_[hash].[ext]', //  占位符 [name]⽼资源模块的名称  [ext]⽼资源模块的后缀
                    outputPath:'images/' //打包后的存放位置
                }
            }
        }
    ]
}
index.js
import pic from "./logo.png";
var img = new Image();
img.src = pic;
img.classList.add("logo");
var root = document.getElementById("root");
root.append(img);

1.2 url-loader (file-loader加强版)

url-loader内部使用了file-loader,所以可以处理file-loader所有的事情,但是遇到图片模块,则会把图片转为base64格式字符串,并打包到js里。这种方式适合小体积的图片。

安装
npm i url-loader -D
配置
module:{
    rules:[
        {
            test:/\.(png|jpe?g|gif)$/,
            use:{
                loader:'url-loader',
                options:{
                    name: '[name]_[hash].[ext]',
                    outputPath:'images/',
                    //⼩于2048,才转换成base64
                    limit:2048
                }
            }
        }
    ]
}

1.3 样式处理

css-loader 处理css模块之间的关系,并合成一个css
style-loader 会把css-loader生成的内容,以style标签挂载到页面的head部分

安装

npm i css-loader style-loader -D

配置

module:{
    rules:[
        {
            test:/\.css$/,
            use:['style-loader','css-loader']
        }
    ]
}
less样式处理

less 把less语法转成css
less-loader less和webpack沟通的桥梁
安装

npm i less less-loader -D

配置

{
    test:/\.less$/,
    use:['style-loader','css-loader','less-loader']
}
样式自动添加前缀

需要用到postcss-loaderautoprefixer
安装

npm i postcss-loader autoprefixer -D

配置

// webpack.config.js
{
    test:/\.css$/,
    use:['style-loader','css-loader','postcss-loader']
}

// postcss.config.js
module.expors = {
    plugins:[
        require('autoprefixer')({
            overrideBrowserslist:['last 2 version', '>1%']
        })
    ]
}

2.如何编写自定义loader

loader 本质就是一个函数,声明式函数,不能用箭头函数
原理是拿到源代码,经过处理后,返回处理后的源码即可。

接下来我们来实现一个简单的loader

  • 创建一个替换源码字符串的loader
//index.js
console.log("hello aaa");

// loader/replaceLoader.js'
module.exports = function(source){
    console.log(source, this, this.query)
    return source.replace('aaa', 'bbb')
}
// 需要⽤声明式函数,因为要上到上下⽂的this,⽤到this的数据,该函数接受⼀个参数,是源码
  • 在配置中使用loader
const path = require('path')
module:{
    rules:[
        test:/\.js$/,
        use:path.resolve(__dirname, './loader/replaceLoader.js')
    ]
}
  • 如何给loader配置参数,loader如何获取参数
    在loader中获取参数有两种方式,this.queryloader-util(需要安装)
// webpack.config.js
{
    test:/\.js$/,
    use:{
        loader:'path.resolve(__dirname, './loader/replaceLoader.js'),
        options:{
            name:'bbb'
        }
    }
}

// loader/replaceLoader.js
const loaderUtils = require("loader-utils");//官⽅推荐处理loader,query的⼯具
module.exports = function(source){
    //this.query 通过this.query来接受配置⽂件传递进来的参数
    // return source.replace('aaa', this.query.name)
    
    const options = loaderUtils.getOptions(this)
    const result = source.replace('aaa', options.name)
    return result
}
  • 如何返回多个信息,不止是处理好的源码呢, (this.callback)
// loader/replaceLoader.js
const loaderUtils = require("loader-utils");//官⽅推荐处理loader,query的⼯具
module.exports = function(source){
    //this.query 通过this.query来接受配置⽂件传递进来的参数
    // return source.replace('aaa', this.query.name)
    
    const options = loaderUtils.getOptions(this)
    const result = source.replace('aaa', options.name)
    this.callback(null, result)
}

//this.callback(
err: Error | null,  
content: string | Buffer,  
sourceMap?: SourceMap,  
meta?: any
);

  • 如何处理异步 (this.async)
const loaderUtils = require("loader-utils");
module.exports = function(source){
    
    const options = loaderUtils.getOptions(this)
    // 定义⼀个异步处理,告诉webpack,这个loader⾥有异步事件,在⾥⾯调⽤下这个异步
    // 使⽤this.asycn来处理,他会返回this.callback
    const callback = this.async()

    setTimeout(()=>{
           const result = source.replace('aaa', options.name)
           callback(null, result)
    },1000)
    
}
  • 处理loader的路径问题
// webpack.config.js
{
    resolveLoader:{
        modules:['node_modules', './loader']
    },
    module:{
        rules:[
            {
                test:/\.js$/,
                use: 'replaceLoader'
            }
        ]
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342