webpack dev server只适合SPA吗

417被浏览27636分享邀请回答/mutualofomaha/multipage-webpack-plugin/tree/master/examples/multiple-nested-entries ), we create ".twig" partials that other ".twig" files consume based on convention. Hopefully this helps, and sorry for speaking in english I currently do not know enough chinese/mandarin to be of use without english.35345 条评论分享收藏感谢收起236 条评论分享收藏感谢收起查看更多回答webpack与SPA实践之管理CSS等资源配置在webpack配置文件的模块加载器选项中添加如下配置:module: {
loaders: [
{ test: /\.css$/, loader: "style-loader!css-loader" }
当然为了方便使用引用路径,还可以配置路径片段别名:alias: {
styles: path.resolve(__dirname, 'src/styles/')
此时,import 'styles/index.css';等同于使用相对路径,如import '../src/styles/indx.css';使用配置好以后,假如我们在styles目录下创建了一个index.css文件,现在可以在JavaScript文件中直接引入该CSS: import 'styles/index.css'; 或 require('styles/index.css');css内容如下:html, body {
width: 100%;
height: 100%;
.container {
页面展示如图:内联样式前面提到了webpack处理CSS的方式有两种,第一种是以内联方式在页面&head&标签内动态插入&style&内联样式,这种方式也是webpack的默认处理方式,只需要简单配置如下加载器:
test: /\.css$/,
exclude: /node_modules/,
loader: 'style-loader!css-loader'
// loaders: ['style-loader', 'css-loader']
WEBPACK加载器解析顺序如上面代码所示,无论是字符串语法style-loader!css-loader,亦或是数组语法['style-loader', 'css-loader'],webpack解析规则都是从右至左,依次解析并执行加载器处理文件,前一加载器处理的输出就是下一加载器处理的输入,直到最后加载器处理完成;此处即webpack先调用css-loader加载器处理css文件,然后将处理结果传递给style-loader加载器,style-loader接受该输入继续处理。CSS-LOADER我们已经反复强调,webpack只能处理JavaScript,所以对于其他诸如css或图片等资源需要使用加载器将其转换输出为JavaScript模块,webpack才能继续处理。css-loader加载器的作用就是支持我们像使用JavaScript模块一样在JavaScript文件中引用CSS文件,如require ('./index.css'),所以你可以认为其作用是将CSS文件转换成JavaScript模块,于是我们可以直接通过引入JavaScript模块的方式直接引用。参数css-loader有两个常用参数:modules: {boolean}指定是否使用CSS模块(如:local和:global设置局部或全局样式规则),默认是false,开启设置如css-loader?importLoaders: {number}指定css-loader加载器之前使用的加载器数量,默认是0,设置如css-loader?importLoaders=1;STYLE-LOADER无论webpack怎么处理CSS文件,最终都需要将其输出到页面,才能实际使用该CSS规则,style-loader加载器就是将CSS以内联方式插入到页面文档,即:针对每一个输入(通过require引入,使用css-loader转换为JavaScript模块,传递给style-loader作为输入),style-loader在页面&head&标签内插入一个&style&标签,该标签内样式规则即此输入内容,如下实例:外链样式当然,我们并不总是希望所有样式都以内联方式存在页面中,很多时候我们也希望通过外链方式使用样式表,特别是样式规则较多的时候,webpack开发者们当然考虑了这样的需求。webpack提供的style-loader加载器默认是以内联方式将样式插入文档,我们需要使用webpack extract-text-webpack-plugin插件以实现输出单独CSS文件。EXTRACT TEXT PLUGIN安装首先安装该插件:
npm install --save-dev extract-text-webpack-plugin
配置然后添加如下配置:
var ExtractTextPlugin = require('extract-text-webpack-plugin');
loaders: [
test: /\.css$/,
exclude: /node_modules/,
// 老版本 loader: ExtractTextPlugin.extract('style-loader', 'css-loader')
loader: ExtractTextPlugin.extract({
fallback:'style-loader',
use: 'css-loader'
plugins: [
// 生成独立css文件
new ExtractTextPlugin({
filename: 'css/[name].css'
运行webpack命令,我们会看到在dist/css/文件夹下生成相应的CSS文件。参数filename {String | Function}Extract Text Plugin为每个入口生成一个CSS文件,所以对于多入口项目需要指定filename参数[name]或[id]或[contenthash]生成唯一识别文件名;disable {Boolean}禁用此插件;allChunks {Boolean}allChunks:时指定从所有模块中抽取CSS输出至单独CSS文件,包括异步引入的额外模块;此插件默认是只抽取初始模块内引入的CSS;extract方法该方法可以以参数指定加载器,然后接受该加载器的输出,进行处理。需要在加载器和插件配置中同时声明相关配置,才能实现效果;在加载器配置中调用其extract方法传入通常以下两个参数:1. use: 将CSS转换成模块的加载器;
2. fallback: 对于不被抽取输出至单独CSS文件的CSS模块使用的加载器,上例中`style-loader`即说明以内联方式使用,该加载器通常在`allChunks: false`时处理额外的模块;
FILENAME与OUTPUT在上一篇介绍了输出文件配置output相关内容,其中:output.path是webpack处理文件后输出的路径,对于CSS文件输出依然适用,即CSS文件也将输出至该目录;output.publicPath是指浏览器访问资源时的相对基础路径,规则是: output.publicPath + output.你可以看到在本系列文章实例中filename都添加了前缀目录,如css和scripts,你可能看到很多项目是不添加的,但文件入口较多时建议分类型目录输出,而且需要记得在浏览器访问资源时也需要添加该目录层级。CSS预处理器通常在开发较复杂的应用时,我们都会选择一种CSS的强化辅助工具,以更高效,更方便的使用CSS开发应用样式,这些拓展工具就是所说的CSS预处理器.CSS预处理器(preprocessors)在CSS语法的基础上增加了变量 (variables)、嵌套 (nested rules)、混合 (mixins)、导入 (inline imports) 等高级功能,令CSS更加强大与优雅,有助于更好地组织管理样式文件,以及更高效地开发项目。目前最常见的CSS预处理器有LESS,SASS,Stylus,个人用过的是前两种,使用SASS的还是居多。SASS安装
npm install --save-dev sass-loader
安装sass-loader以后会发现,package.json中多了一个node-sass依赖,这是使用SASS必须的。配置然后添加以下配置:
test: /\.s[ac]ss$/,
exclude: /node_modules/,
loader: 'style-loader!css-loader!sass-loader'
如上,配置中传递了三个加载器,相对于前文处理CSS文件的加载器,在最后面多了一个sass-loader,首先加载sass-loader加载器处理SASS文件成CSS代码,然后继续按照前文描述流程处理CSS文件。EXTRACT TEXT PLUGIN和处理CSS文件一样,上述配置最终通过style-loader将转换后的CSS代码内联到页面,我们需要使用Extract Text Plugin生成单独CSS文件,以外链方式引用:
test: /\.s[ac]ss$/,
exclude: /node_modules/,
loader: ExtractTextPlugin.extract({
fallback:'style-loader',
'css-loader',
'sass-loader'
// 生成独立css文件
new ExtractTextPlugin({
filename: 'css/[name].css'
CSS后处理器前面讲到CSS预处理器,如SASS,他们提供对CSS的拓展,包括语法拓展,高级特性拓展,如嵌套,变量,自动处理添加属性前缀等,使得我们可以以其定义的语法与模板方式更高效的编写CSS,然而这些预处理器都是另外对CSS进行拓展,各自定义了语法和模板,其处理流程是对代码进行解析处理,然后转换成CSS代码。不同预处理器有各自的定义和规范,假如你需要从LESS转到SASS,源代码转换成本和学习成本颇高,而接下来要介绍的CSS后处理器并没有这个问题。不同于预处理器预定义好一个语法和模板,然后对按照该语法和模板编写的代码进行处理转换成CSS,其输入是自定义语法文件,输出是CSS代码;后处理器(postprocessor)是对原生CSS代码根据配置进行处理,其输入输出依然是CSS代码。POSTCSS现在最受欢迎的CSS后处理器,就是postcss:PostCSS is a tool for transforming styles with JS plugins. These plugins can lint your CSS, support variables and mixins, transpile future CSS syntax, inline images, and more.PostCSS是一个使用Js插件转换样式的根据,插件支持拓展CSS,如变量,混合,CSS属性语法兼容,行内图片等等功能。特性不同于SASS提供一个功能性拓展工具,postcss更多的是提供一个CSS高效开发工具解决方式,其本身只包含CSS解析器只能将CSS处理成一棵抽象语法树(AST),同时提供一个丰富的CSS节点树API,可以对语法树进行操作,另外它有一个高拓展性的插件系统支持我们通过引入不同插件对CSS进行处理,一个插件的输出还可以是下一个插件的输入,更值得一提的是,这些插件都是JavaScript插件,前端开发者们很容易就能根据项目需求定制自己的插件,所以可以总结几点一以下特性:postcss只处理CSS,概念简洁;提供高拓展性的插件系统支持按需引入不同插件,实现不同处理;使用JavaScript插件,开发者可以很方便定制项目插件;提供CSS节点树API,可以高效的进行CSS处理;安装在webpack中使用,需要先安装对应加载器:
npm install --save-dev postcss-loader
插件postcss目前有200+插件,足够满足绝大部分项目开发需求,可以查看,我们介绍几个主要使用的插件。Autoprefixer回顾一下在预处理器中,如果我们需要为CSS代码添加属性前缀,需要这么实现呢?对于Sass,我们通常使用mixin,即混合宏,处理CSS属性前缀,如:// 定义
@mixin prefix-animation($animation-name){
animation:$animation-
-webkit-animation:$animation-
@include prefix-animation(loading .5s linear infinite);
如上,我们需要按照定义的语法和模板:先定义一个mixin,然后通过@include方式使用,最后才能输出添加前缀的CSS代码,当代码越来越多时,我们需要定义的mixin也会越来越多,而且不同预处理器定义的语法和模板都有差异,学习成本、转换成本都很可能令人难以接受。那么postcss插件怎么处理的呢?postcss提供了Autoprefixer插件处理CSS属性前缀:Autoprefixer插件基于的数据,对CSS规则进行前缀处理。安装首先还是要安装Autoprefixer:
npm install --save-dev autoprefixer
配置添加如下配置:module: {
loaders: [
test: /\.css$/,
exclude: /node_modules/,
loaders: [
'style-loader',
'css-loader',
loader: 'postcss-loader',
options: {
plugins: [
require('autoprefixer')({
browsers: ['last 2 versions']
如上,我们知道postcss是一个样式开发解决方案,其特定功能需要引入插件实现,上例中在指定postcss-loader加载器时为其设置了插件配置autoprefixer;当然webpack还支持直接设置一个postcss配置文件,然后在项目根目录创建postcss.config.js配置文件,内容格式如下:module.exports = {
plugins: [
require('autoprefixer')({
browsers: ['last 2 versions']
// or just require('autoprefixer')
使用autoprefixer插件时可选传入browsers参数,可以设置添加前缀的适配范围,详细可查阅。混合使用CSS预处理器与后处理器 – PreCSS也许你迫不及待想在项目中引入postcss,又希望能继续使用CSS预处理器语法,而且需要保证以前按照某预处理器预定语法和模板(如SASS)编写的源代码继续稳定使用,不需要太多的迁移和学习成本,可以做到吗?当然可以,可以使用预处理器PreCSS插件包,另外我们需要安装一个postcss的scss解析器,因为postcss默认只包含一个CSS解析器,postcss配置文件更新如下:module.exports = {
parser: require('postcss-scss'),
plugins: [
require('autoprefixer')({
browsers: ['last 2 versions']
require('precss')
webpack配置文件更新配置:modules: {
loaders: [
test: /\.s?[ac]ss$/,
exclude: /node_modules/,
// or 内联方式 loader: 'style-loader!css-loader!postcss-loader'
loader: ExtractTextPlugin.extract({
fallback:'style-loader',
'css-loader',
'postcss-loader'
可以看到文件匹配规则,修改为/\.s?[ac]ss$/,可以匹配包括.sass, .scss, .css样式文件;在css-loader加载器之前添加了postcss-loader加载器(webpack加载器解析顺序为从右至左)。当然你可以不使用precss,依然使用sass-loader,则只需要修改配置:
loader: 'style-loader!css-loader!postcss-loader!sass-loader'
对于如下SCSS代码:
$column: 200
width: calc(4 * $column);
转换生成如下CSS代码:.menu {
display: -webkit-
display: -ms-
width: calc(4 * 200px);
处理图片与字体文件对于一个应用而言,除了需要开发HTML、CSS、JavaScript,通常还好使用到图片,字体文件,配置文件等诸多资源,那么前端工程化流程也就必然需要对这些资源进行处理与优化,最典型的说处理图片和字体文件。在Grunt或Gulp中,我们对图片和字体文件的处理通常是将其从源目录压缩优化处理后输出至输出目录,通常是以文件目录整体进行处理,每次构建时,对所有资源,包括未使用的图片均进行处理,效率是有局限的;而webpack中一切资源文件都可以处理成模块,然后在编译时管理模块依赖,可以做到只处理存在依赖的资源(即,使用了的资源)。图片与字体当我们在Js模块中引入CSS文件时,其中样式规则中的背景图片,字体文件如何处理呢?webpack只能管理模块化的东西,需要将其模块化,然后使用webpack管理依赖,webpack提供了file-loader加载器:File LoaderInstructs webpack to emit the required object as file and to return its public url.通知webpack将引入的对象输出为文件并返回其公开资源路径。配置module: {
loaders: [
test: /\.(png|svg|jpe?g|gif)$/,
'file-loader'
说明当我们在js文件中import Image from '../images/test.png'或在CSS文件中url('../images/test.png')时,file-loader将处理该图片并在output.path目录下输出文件,然后将../images/test.png路径替换成该输出文件路径。注,对于html中引用的图片,需要使用[html-loader]加载器处理()。参数emitFile: 是否输出文件;name: 指定输出文件的文件名,有几个可用内置变量:[name]: 引用资源的名称;[path]: 引用资源的相对路径;[ext]: 资源拓展名;[hash]: 资源内容的hash值,默认使用md5算法计算得到,可以指定长度值,如[hash:7]表示返回hash值前7个字符;[hashType:hash:digestType:length]: 指定hash值计算算法类型和摘要类型,及摘要长度,如sha512:hash:base64:7表示使用sha512加密算法计算hash值并且返回7个字符的base64编码字符实例在配置时可以指定参数:file-loader?name=[name].[ext]?[hash:8]或者以配置对象方式:
test: /\.(png|svg|jpe?g|gif)$/,
loaders: [
// 'file-loader?name=[path][name].[ext]?[hash:8]'
loader: 'file-loader',
name: '[path][name].[ext]?[hash:8]'
对于CSS源代码:.wrapper {
font-size: 18
background: url('../images/test.png') no-repeat 0 0;
输出CSS代码如下:.wrapper {
font-size: 18
background: url(assets/images/test.png?) no-repeat 0 0;
assets为output.publicPath指定值,images/test.png?为配置文件中指定的name模板,在output.path目录下输出images/test.png,区别是,不会携带?后的参数。另外,你也可以在js模板中这样使用:
&img src={imgSrc} /&
import imgSrc from 'path/xxx.png';
Url Loader你可能会发现前面并没有安装file-loader,因为有更好用的加载器url-loader,url-loader加载器是file-loader的升级版,他们唯一的不同之处在于:url-loader可以通过limit参数指定一个尺寸值,加载器会对小于该值的资源处理返回一个Data URL,以base64的方式嵌入HTML或CSS,如url-loader?limit=65000;对于大于该尺寸的资源将使用file-loader处理并且传递所有参数。mimetype还可以设置mimetype对处理文件进行过滤,如url-loader?mimetype=image/png将只处理png类型的资源。安装
npm install --save-dev url-loader
配置该加载器对于图片和字体文件资源都适用:
test: /\.(png|svg|jpe?g|gif)$/,
loaders: [
// 'url-loader?name=[path][name].[ext]?[hash:8]'
loader: 'url-loader',
limit: 6000,
name: '[path][name].[ext]?[hash:8]'
test: /\.(woff|woff2|eot|ttf|otf)$/,
loaders: [{
loader: 'url-loader',
limit: 10000,
name: '[path][name].[ext]?[hash:8]'
资源优化完成以上配置后,已经可以在项目中很方便的引用各自资源了,但是通常我们还需要对图片字体等文件进行压缩优化处理,如Grunt中使用的imagemin插件一样压缩资源,webpack则提供了相关加载器img-loader。安装
npm install --save-dev img-loader
test: /\.(jpe?g|png|gif|svg)$/i,
loaders: [
'url-loader?name=[path][name].[ext]?[hash:8]',
loader: 'img-loader',
options: {
// 根据环境判断是否启用资源压缩
enabled: process.env.NODE_ENV === 'production',
gifsicle: {
interlaced: false // 替换使用渐进式渲染
mozjpeg: {
progressive: true, // 创建基准jpeg文件
optipng: {
optimizationLevel: 4, // 优化级别,0-7,值越大,压缩越多
pngquant: {
quality: '75-90', // 压缩质量,0-100,值越高,质量越高
speed: 3 // 执行速度,0-10,速度过高质量受损,不建议过高
plugins: [
{ removeTitle: true }, // 去除标题信息
{ convertPathData: false } // 转换路径为更短的相对或决定路径
以上为常见使用配置,更多详细配置信息请查看对应说明,特别注意的是上面使用了process.env.NODE_ENV当前环境变量,只有在生产环境启用图片压缩,因为压缩过程比较比较耗时,可能会降低开发、调试效率。数据资源对于数据类型文件资源,webpack内置支持加载解析.json文件,而其他类型则需要安装配置相应加载器,如.xml文件,需要安装并配置xml-loader。资源管理的思考在传统或稍早一点的应用中,我们通常会将所有的图片,字体等资源放在一个基础目录下,如assets/或images,但是对于那些在多项目间重复的插件代码或资源来说,每一次迁移,我们都得在一大堆图片,字体资源里寻找出我们需要迁移的资源,这对代码可重用和其独立性有一定限制,而且与现在提倡的组件化开发模式也不相符。webpack对于资源的处理方式给组件化开发提供了很大便利,使得我们以组件为单位,可以在某一组件目录下存放所有相关的js,css,图片,字体等资源文件;组件的迁移公用成本很低。不过组件化开发并不是说不需要资源目录了,一些公用的资源依然放在项目的基础目录下。说明终于可以松口气,对于webpack管理CSS、图片、字体、数据资源的实践基本总结完成,其实感觉要介绍的还有很多,但是要尽量保证文章思路清晰,语句流畅,而且篇幅不能太长,水平有限,花费较多时间经历,希望能对读者有所帮助,后续篇章也会继续穿插介绍,力争本系列能较完整、较清晰地描述如何使用webpack开发SPA应用。原创文章,转载请注明: 转载自 本文链接地址: 赞赏还没有人赞赏,快来当第一个赞赏的人吧!18收藏分享举报文章被以下专栏收录高效的中文IT技术平台推荐阅读{&debug&:false,&apiRoot&:&&,&paySDK&:&https:\u002F\\u002Fapi\u002Fjs&,&wechatConfigAPI&:&\u002Fapi\u002Fwechat\u002Fjssdkconfig&,&name&:&production&,&instance&:&column&,&tokens&:{&X-XSRF-TOKEN&:null,&X-UDID&:null,&Authorization&:&oauth c3cef7c66aa9e6a1e3160e20&}}{&database&:{&Post&:{&&:{&isPending&:false,&contributes&:[{&sourceColumn&:{&lastUpdated&:,&description&:&一个专注互联网相关技术的专栏&,&permission&:&COLUMN_PUBLIC&,&memberId&:,&contributePermission&:&COLUMN_PUBLIC&,&translatedCommentPermission&:&all&,&canManage&:true,&intro&:&高效的中文IT技术平台&,&urlToken&:&dreawer&,&id&:21254,&imagePath&:&v2-def9c21d9ca33ad157f4208.jpg&,&slug&:&dreawer&,&applyReason&:&0&,&name&:&极乐科技&,&title&:&极乐科技&,&url&:&https:\u002F\\u002Fdreawer&,&commentPermission&:&COLUMN_ALL_CAN_COMMENT&,&canPost&:true,&created&:,&state&:&COLUMN_NORMAL&,&followers&:25154,&avatar&:{&id&:&v2-def9c21d9ca33ad157f4208&,&template&:&https:\u002F\\u002F{id}_{size}.jpg&},&activateAuthorRequested&:false,&following&:false,&imageUrl&:&https:\u002F\\u002Fv2-def9c21d9ca33ad157f4208_l.jpg&,&articlesCount&:554},&state&:&accepted&,&targetPost&:{&titleImage&:&&,&lastUpdated&:,&imagePath&:&&,&permission&:&ARTICLE_PUBLIC&,&topics&:[036,369],&summary&:&\u003Ca href=\&http:\u002F\\u002F\u002F31\u002Fwebpack_spa_build_env\u002F\& data-editable=\&true\& data-title=\&上一篇\& class=\&\&\u003E上一篇\u003C\u002Fa\u003E介绍了如何使用webpack搭建一个稳定的支持本地服务、自动刷新、模块热替换、使用ES6编写JavaScript的开发环境,本篇主要介绍webpack如何处理HTML应用三大元素的另一元素 – CSS及其他诸如图片、字体文件或者数据配置文件等资源。前言 在学习使用webpa…&,&copyPermission&:&ARTICLE_COPYABLE&,&translatedCommentPermission&:&all&,&likes&:0,&origAuthorId&:0,&publishedTime&:&T11:53:41+08:00&,&sourceUrl&:&&,&urlToken&:,&id&:3101231,&withContent&:false,&slug&:,&bigTitleImage&:false,&title&:&webpack与SPA实践之管理CSS等资源&,&url&:&\u002Fp\u002F&,&commentPermission&:&ARTICLE_ALL_CAN_COMMENT&,&snapshotUrl&:&&,&created&:,&comments&:0,&columnId&:21254,&content&:&&,&parentId&:0,&state&:&ARTICLE_PUBLISHED&,&imageUrl&:&&,&author&:{&bio&:&前端开发工程师,15年毕业于北京交通大学,爱好前端,喜好尝试新技术,热爱分享。&,&isFollowing&:false,&hash&:&e0a0cf37d23&,&uid&:368300,&isOrg&:false,&slug&:&codingplayboy&,&isFollowed&:false,&description&:&熊建刚,前端开发工程师,2015年毕业于北京交通大学,爱好前端,喜好尝试新技术,热爱分享。\n技能:\n– 前端技能:精通HTML5,CSS3,JavaScript,jQuery,BackBone,ReactJs,Vue.js,熟悉AngularJs 1.x,underscore\n– 自动化构建:熟练掌握Grunt,熟悉webpack,babel,了解gulp,browserify\n– 模块化:熟悉cmd ,amd, umd\n– 后端:熟悉NodeJs, Java\n– 数据库:MySQL,MongoDB\n– 版本工具:熟练掌握Git,SVN\n– 其他:微信公众号NodeJs开发(服务号、企业号开发)&,&name&:&熊建刚&,&profileUrl&:&https:\u002F\\u002Fpeople\u002Fcodingplayboy&,&avatar&:{&id&:&da8e974dc&,&template&:&https:\u002F\\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false},&memberId&:,&excerptTitle&:&&,&voteType&:&ARTICLE_VOTE_CLEAR&},&id&:673276}],&title&:&webpack与SPA实践之管理CSS等资源&,&author&:&codingplayboy&,&content&:&\u003Cp\u003E\u003Ca href=\&http:\u002F\\u002F?target=http%3A\u002F\\u002F\u002F31\u002Fwebpack_spa_build_env\u002F\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003E上一篇\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E介绍了如何使用webpack搭建一个稳定的支持本地服务、自动刷新、模块热替换、使用ES6编写JavaScript的开发环境,本篇主要介绍webpack如何处理HTML应用三大元素的另一元素 – CSS及其他诸如图片、字体文件或者数据配置文件等资源。\u003C\u002Fp\u003E\u003Cp\u003E前言\u003Cbr\u003E\u003C\u002Fp\u003E\u003Cp\u003E在学习使用webpack时,我们需要明白无论它怎么设计,它的工作原理、流程是什么,最根本的它处理的还是HTML文档中的HTML标签、JavaScript、CSS、图片等资源,而且最终的处理结果依然必须是一个HTML文档,包括DOM、JavaScript、CSS,而CSS在文档中的存在方式,有三种:行内样式,内联样式,外链样式,行内样式使用方式早已不推荐,所以webpack处理CSS方式也就两种:\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E内联样式: 以&style&标签方式在HTML文档中嵌入样式;\u003C\u002Fli\u003E\u003Cli\u003E外链样式: 打包生成CSS文件,通过&link&标签引入样式;\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Ch2\u003Ewebpack与CSS\u003C\u002Fh2\u003E\u003Cp\u003E我们知道,webpack本质是只能处理JavaScript的,而对于其他资源,需要使用加载器和插件将其处理成JavaScript模块,然后进行模块依赖管理。webpack提供style-loader和css-loader两个加载器支持我们模块化CSS,因此可以在其他模块内直接引入。\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev style-loader css-loader\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E在webpack配置文件的模块加载器选项中添加如下配置:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Emodule: {\n
loaders: [\n
{ test: \u002F\\.css$\u002F, loader: \&style-loader!css-loader\& }\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E当然为了方便使用引用路径,还可以配置路径片段别名:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Ealias: {\n
styles: path.resolve(__dirname, 'src\u002Fstyles\u002F')\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E此时,import 'styles\u002Findex.css';等同于使用相对路径,如import '..\u002Fsrc\u002Fstyles\u002Findx.css';\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E使用\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E配置好以后,假如我们在styles目录下创建了一个index.css文件,现在可以在JavaScript文件中直接引入该CSS: import 'styles\u002Findex.css'; 或 require('styles\u002Findex.css');\u003C\u002Fp\u003E\u003Cp\u003Ecss内容如下:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Ehtml, body {\n
width: 100%;\n
height: 100%;\n
.container {\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E页面展示如图:\u003Cnoscript\u003E\u003Cimg src=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_b.png\& data-rawwidth=\&433\& data-rawheight=\&272\& class=\&origin_image zh-lightbox-thumb\& width=\&433\& data-original=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_r.png\&\u003E\u003C\u002Fnoscript\u003E\u003Cimg src=\&data:image\u002Fsvg+utf8,&svg%20xmlns='http:\u002F\u002Fwww.w3.org\u002FFsvg'%20width='433'%20height='272'&&\u002Fsvg&\& data-rawwidth=\&433\& data-rawheight=\&272\& class=\&origin_image zh-lightbox-thumb lazy\& width=\&433\& data-original=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_r.png\& data-actualsrc=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_b.png\&\u003E\u003C\u002Fp\u003E\u003Ch2\u003E内联样式\u003C\u002Fh2\u003E\u003Cp\u003E前面提到了webpack处理CSS的方式有两种,第一种是以内联方式在页面&head&标签内动态插入&style&内联样式,这种方式也是webpack的默认处理方式,只需要简单配置如下加载器:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
test: \u002F\\.css$\u002F,\n
exclude: \u002Fnode_modules\u002F,\n
loader: 'style-loader!css-loader'\n
\u002F\u002F or \n
\u002F\u002F loaders: ['style-loader', 'css-loader']\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Ch4\u003EWEBPACK加载器解析顺序\u003C\u002Fh4\u003E\u003Cp\u003E如上面代码所示,无论是字符串语法style-loader!css-loader,亦或是数组语法['style-loader', 'css-loader'],webpack解析规则都是从右至左,依次解析并执行加载器处理文件,前一加载器处理的输出就是下一加载器处理的输入,直到最后加载器处理完成;此处即webpack先调用css-loader加载器处理css文件,然后将处理结果传递给style-loader加载器,style-loader接受该输入继续处理。\u003C\u002Fp\u003E\u003Ch4\u003ECSS-LOADER\u003C\u002Fh4\u003E\u003Cblockquote\u003E\u003Cp\u003E我们已经反复强调,webpack只能处理JavaScript,所以对于其他诸如css或图片等资源需要使用加载器将其转换输出为JavaScript模块,webpack才能继续处理。\u003C\u002Fp\u003E\u003C\u002Fblockquote\u003E\u003Cp\u003Ecss-loader加载器的作用就是支持我们像使用JavaScript模块一样在JavaScript文件中引用CSS文件,如require ('.\u002Findex.css'),所以你可以认为其作用是将CSS文件转换成JavaScript模块,于是我们可以直接通过引入JavaScript模块的方式直接引用。\u003C\u002Fp\u003E参数\u003Cp\u003Ecss-loader有两个常用参数:\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003Emodules: {boolean}指定是否使用CSS模块(如:local和:global设置局部或全局样式规则),默认是false,开启设置如css-loader?\u003C\u002Fli\u003E\u003Cli\u003EimportLoaders: {number}指定css-loader加载器之前使用的加载器数量,默认是0,设置如css-loader?importLoaders=1;\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Ch4\u003ESTYLE-LOADER\u003C\u002Fh4\u003E\u003Cp\u003E无论webpack怎么处理CSS文件,最终都需要将其输出到页面,才能实际使用该CSS规则,style-loader加载器就是将CSS以内联方式插入到页面文档,即:针对每一个输入(通过require引入,使用css-loader转换为JavaScript模块,传递给style-loader作为输入),style-loader在页面&head&标签内插入一个&style&标签,该标签内样式规则即此输入内容,如下实例:\u003Cnoscript\u003E\u003Cimg src=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_b.png\& data-rawwidth=\&433\& data-rawheight=\&272\& class=\&origin_image zh-lightbox-thumb\& width=\&433\& data-original=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_r.png\&\u003E\u003C\u002Fnoscript\u003E\u003Cimg src=\&data:image\u002Fsvg+utf8,&svg%20xmlns='http:\u002F\u002Fwww.w3.org\u002FFsvg'%20width='433'%20height='272'&&\u002Fsvg&\& data-rawwidth=\&433\& data-rawheight=\&272\& class=\&origin_image zh-lightbox-thumb lazy\& width=\&433\& data-original=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_r.png\& data-actualsrc=\&https:\u002F\\u002Fv2-e8aafccf395a62a24bbdeb4_b.png\&\u003E\u003C\u002Fp\u003E\u003Ch2\u003E外链样式\u003C\u002Fh2\u003E\u003Cp\u003E当然,我们并不总是希望所有样式都以内联方式存在页面中,很多时候我们也希望通过外链方式使用样式表,特别是样式规则较多的时候,webpack开发者们当然考虑了这样的需求。\u003C\u002Fp\u003E\u003Cp\u003Ewebpack提供的style-loader加载器默认是以内联方式将样式插入文档,我们需要使用webpack extract-text-webpack-plugin插件以实现输出单独CSS文件。\u003C\u002Fp\u003E\u003Ch4\u003EEXTRACT TEXT PLUGIN\u003C\u002Fh4\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E首先安装该插件:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev extract-text-webpack-plugin\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E然后添加如下配置:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
var ExtractTextPlugin = require('extract-text-webpack-plugin');\n
module: {\n
loaders: [\n
test: \u002F\\.css$\u002F,\n
exclude: \u002Fnode_modules\u002F,\n
\u002F\u002F 老版本 loader: ExtractTextPlugin.extract('style-loader', 'css-loader')\n
loader: ExtractTextPlugin.extract({\n
fallback:'style-loader',\n
use: 'css-loader'\n
plugins: [\n
\u002F\u002F 生成独立css文件\n
new ExtractTextPlugin({\n
filename: 'css\u002F[name].css'\n
]\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E运行webpack命令,我们会看到在dist\u002Fcss\u002F文件夹下生成相应的CSS文件。\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E\u003Cp\u003E参数\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E\u003Cp\u003Efilename {String | Function}\u003C\u002Fp\u003E\u003Cp\u003EExtract Text Plugin为每个入口生成一个CSS文件,所以对于多入口项目需要指定filename参数[name]或[id]或[contenthash]生成唯一识别文件名;\u003C\u002Fp\u003E\u003C\u002Fli\u003E\u003Cli\u003E\u003Cp\u003Edisable {Boolean}\u003C\u002Fp\u003E\u003Cp\u003E禁用此插件;\u003C\u002Fp\u003E\u003C\u002Fli\u003E\u003Cli\u003E\u003Cp\u003EallChunks {Boolean}\u003C\u002Fp\u003E\u003Cp\u003EallChunks:时指定从所有模块中抽取CSS输出至单独CSS文件,包括异步引入的额外模块;此插件默认是只抽取初始模块内引入的CSS;\u003C\u002Fp\u003E\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003C\u002Fli\u003E\u003Cli\u003E\u003Cp\u003Eextract方法\u003C\u002Fp\u003E\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E该方法可以以参数指定加载器,然后接受该加载器的输出,进行处理。需要在加载器和插件配置中同时声明相关配置,才能实现效果;在加载器配置中调用其extract方法传入通常以下两个参数:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E1. use: 将CSS转换成模块的加载器;\n2. fallback: 对于不被抽取输出至单独CSS文件的CSS模块使用的加载器,上例中`style-loader`即说明以内联方式使用,该加载器通常在`allChunks: false`时处理额外的模块;\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Ch4\u003EFILENAME与OUTPUT\u003C\u002Fh4\u003E\u003Cp\u003E在上一篇介绍了输出文件配置output相关内容,其中:\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003Eoutput.path是webpack处理文件后输出的路径,对于CSS文件输出依然适用,即CSS文件也将输出至该目录;\u003C\u002Fli\u003E\u003Cli\u003Eoutput.publicPath是指浏览器访问资源时的相对基础路径,规则是: output.publicPath + output.\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E\u003Cem\u003E你可以看到在本系列文章实例中filename都添加了前缀目录,如css和scripts,你可能看到很多项目是不添加的,但文件入口较多时建议分类型目录输出,而且需要记得在浏览器访问资源时也需要添加该目录层级。\u003C\u002Fem\u003E\u003C\u002Fp\u003E\u003Ch2\u003ECSS预处理器\u003C\u002Fh2\u003E\u003Cp\u003E通常在开发较复杂的应用时,我们都会选择一种CSS的强化辅助工具,以更高效,更方便的使用CSS开发应用样式,这些拓展工具就是所说的CSS预处理器.\u003C\u002Fp\u003E\u003Cblockquote\u003E\u003Cp\u003E\u003Cstrong\u003ECSS预处理器(preprocessors)\u003C\u002Fstrong\u003E在CSS语法的基础上增加了变量 (variables)、嵌套 (nested rules)、混合 (mixins)、导入 (inline imports) 等高级功能,令CSS更加强大与优雅,有助于更好地组织管理样式文件,以及更高效地开发项目。\u003C\u002Fp\u003E\u003C\u002Fblockquote\u003E\u003Cp\u003E目前最常见的CSS预处理器有LESS,SASS,Stylus,个人用过的是前两种,使用SASS的还是居多。\u003C\u002Fp\u003E\u003Ch4\u003ESASS\u003C\u002Fh4\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev sass-loader\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E\u003Cem\u003E安装sass-loader以后会发现,package.json中多了一个node-sass依赖,这是使用SASS必须的。\u003C\u002Fem\u003E\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E然后添加以下配置:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
test: \u002F\\.s[ac]ss$\u002F,\n
exclude: \u002Fnode_modules\u002F,\n
loader: 'style-loader!css-loader!sass-loader'\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E如上,配置中传递了三个加载器,相对于前文处理CSS文件的加载器,在最后面多了一个sass-loader,首先加载sass-loader加载器处理SASS文件成CSS代码,然后继续按照前文描述流程处理CSS文件。\u003C\u002Fp\u003E\u003Ch4\u003EEXTRACT TEXT PLUGIN\u003C\u002Fh4\u003E\u003Cp\u003E和处理CSS文件一样,上述配置最终通过style-loader将转换后的CSS代码内联到页面,我们需要使用Extract Text Plugin生成单独CSS文件,以外链方式引用:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
test: \u002F\\.s[ac]ss$\u002F,\n
exclude: \u002Fnode_modules\u002F,\n
loader: ExtractTextPlugin.extract({\n
fallback:'style-loader',\n
'css-loader',\n
'sass-loader'\n
\u002F\u002F 生成独立css文件\n
new ExtractTextPlugin({\n
filename: 'css\u002F[name].css'\n
})\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Ch2\u003ECSS后处理器\u003C\u002Fh2\u003E\u003Cp\u003E前面讲到CSS预处理器,如SASS,他们提供对CSS的拓展,包括语法拓展,高级特性拓展,如嵌套,变量,自动处理添加属性前缀等,使得我们可以以其定义的语法与模板方式更高效的编写CSS,然而这些预处理器都是另外对CSS进行拓展,各自定义了语法和模板,其处理流程是对代码进行解析处理,然后转换成CSS代码。\u003C\u002Fp\u003E\u003Cp\u003E不同预处理器有各自的定义和规范,假如你需要从LESS转到SASS,源代码转换成本和学习成本颇高,而接下来要介绍的CSS后处理器并没有这个问题。\u003C\u002Fp\u003E\u003Cblockquote\u003E\u003Cp\u003E不同于预处理器预定义好一个语法和模板,然后对按照该语法和模板编写的代码进行处理转换成CSS,其输入是自定义语法文件,输出是CSS代码;\u003Cstrong\u003E后处理器(postprocessor)\u003C\u002Fstrong\u003E是对原生CSS代码根据配置进行处理,其输入输出依然是CSS代码。\u003C\u002Fp\u003E\u003C\u002Fblockquote\u003E\u003Ch4\u003EPOSTCSS\u003C\u002Fh4\u003E\u003Cp\u003E现在最受欢迎的CSS后处理器,就是postcss:\u003C\u002Fp\u003E\u003Cblockquote\u003E\u003Cp\u003EPostCSS is a tool for transforming styles with JS plugins. These plugins can lint your CSS, support variables and mixins, transpile future CSS syntax, inline images, and more.\u003Cbr\u003EPostCSS是一个使用Js插件转换样式的根据,插件支持拓展CSS,如变量,混合,CSS属性语法兼容,行内图片等等功能。\u003C\u002Fp\u003E\u003C\u002Fblockquote\u003E特性\u003Cp\u003E不同于SASS提供一个功能性拓展工具,postcss更多的是提供一个CSS高效开发工具解决方式,其本身只包含CSS解析器只能将CSS处理成一棵抽象语法树(AST),同时提供一个丰富的CSS节点树API,可以对语法树进行操作,另外它有一个高拓展性的插件系统支持我们通过引入不同插件对CSS进行处理,一个插件的输出还可以是下一个插件的输入,更值得一提的是,这些插件都是JavaScript插件,前端开发者们很容易就能根据项目需求定制自己的插件,所以可以总结几点一以下特性:\u003C\u002Fp\u003E\u003Col\u003E\u003Cli\u003Epostcss只处理CSS,概念简洁;\u003C\u002Fli\u003E\u003Cli\u003E提供高拓展性的插件系统支持按需引入不同插件,实现不同处理;\u003C\u002Fli\u003E\u003Cli\u003E使用JavaScript插件,开发者可以很方便定制项目插件;\u003C\u002Fli\u003E\u003Cli\u003E提供CSS节点树API,可以高效的进行CSS处理;\u003C\u002Fli\u003E\u003C\u002Fol\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E在webpack中使用,需要先安装对应加载器:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev postcss-loader\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E插件\u003Cp\u003Epostcss目前有200+插件,足够满足绝大部分项目开发需求,可以查看\u003Ca href=\&http:\u002F\\u002F?target=https%3A\\u002Fpostcss\u002Fpostcss%23plugins\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003Epostcss插件\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E,我们介绍几个主要使用的插件。\u003C\u002Fp\u003EAutoprefixer\u003Cp\u003E回顾一下在预处理器中,如果我们需要为CSS代码添加属性前缀,需要这么实现呢?对于Sass,我们通常使用mixin,即混合宏,处理CSS属性前缀,如:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E\u002F\u002F 定义\n
@mixin prefix-animation($animation-name){\n
animation:$animation-\n
-webkit-animation:$animation-\n
\u002F\u002F 使用\n
@include prefix-animation(loading .5s linear infinite);\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E如上,我们需要按照定义的语法和模板:先定义一个mixin,然后通过@include方式使用,最后才能输出添加前缀的CSS代码,当代码越来越多时,我们需要定义的mixin也会越来越多,而且不同预处理器定义的语法和模板都有差异,学习成本、转换成本都很可能令人难以接受。\u003C\u002Fp\u003E\u003Cp\u003E那么postcss插件怎么处理的呢?postcss提供了Autoprefixer插件处理CSS属性前缀:\u003C\u002Fp\u003E\u003Cblockquote\u003E\u003Cp\u003EAutoprefixer插件基于\u003Ca href=\&http:\u002F\\u002F?target=http%3A\\u002F\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003ECan I Use\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E的数据,对CSS规则进行前缀处理。\u003C\u002Fp\u003E\u003C\u002Fblockquote\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E首先还是要安装Autoprefixer:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev autoprefixer\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E添加如下配置:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Emodule: {\n
loaders: [\n
test: \u002F\\.css$\u002F,\n
exclude: \u002Fnode_modules\u002F,\n
loaders: [\n
'style-loader',\n
'css-loader',\n
loader: 'postcss-loader', \n
options: {\n
plugins: [\n
require('autoprefixer')({\n
browsers: ['last 2 versions']\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E如上,我们知道postcss是一个样式开发解决方案,其特定功能需要引入插件实现,上例中在指定postcss-loader加载器时为其设置了插件配置autoprefixer;当然webpack还支持直接设置一个postcss配置文件,然后在项目根目录创建postcss.config.js配置文件,内容格式如下:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Emodule.exports = {\n
plugins: [\n
require('autoprefixer')({\n
browsers: ['last 2 versions']\n
\u002F\u002F or just require('autoprefixer')\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E\u003Cem\u003E使用autoprefixer插件时可选传入browsers参数,可以设置添加前缀的适配范围,详细可查阅\u003Ca href=\&http:\u002F\\u002F?target=https%3A\\u002Fpostcss\u002Fautoprefixer%23browsers\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003Ebrowsers配置说明\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E。\u003C\u002Fem\u003E\u003C\u002Fp\u003E混合使用CSS预处理器与后处理器 – PreCSS\u003Cp\u003E也许你迫不及待想在项目中引入postcss,又希望能继续使用CSS预处理器语法,而且需要保证以前按照某预处理器预定语法和模板(如SASS)编写的源代码继续稳定使用,不需要太多的迁移和学习成本,可以做到吗?当然可以,可以使用预处理器PreCSS插件包,另外我们需要安装一个postcss的scss解析器,因为postcss默认只包含一个CSS解析器,postcss配置文件更新如下:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Emodule.exports = {\n
parser: require('postcss-scss'),\n
plugins: [\n
require('autoprefixer')({\n
browsers: ['last 2 versions']\n
require('precss')\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003Ewebpack配置文件更新配置:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Emodules: {\n
loaders: [\n
test: \u002F\\.s?[ac]ss$\u002F,\n
exclude: \u002Fnode_modules\u002F,\n
\u002F\u002F or 内联方式 loader: 'style-loader!css-loader!postcss-loader'\n
loader: ExtractTextPlugin.extract({\n
fallback:'style-loader',\n
'css-loader',\n
'postcss-loader'\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E可以看到文件匹配规则,修改为\u002F\\.s?[ac]ss$\u002F,可以匹配包括.sass, .scss, .css样式文件;在css-loader加载器之前添加了postcss-loader加载器(webpack加载器解析顺序为从右至左)。\u003C\u002Fp\u003E\u003Cp\u003E当然你可以不使用precss,依然使用sass-loader,则只需要修改配置:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
loader: 'style-loader!css-loader!postcss-loader!sass-loader'\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E对于如下SCSS代码:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
$column: 200\n
display:\n
width: calc(4 * $column);\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E转换生成如下CSS代码:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E.menu {\n
display: -webkit-\n
display: -ms-\n
display: \n
width: calc(4 * 200px);\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Ch2\u003E处理图片与字体文件\u003C\u002Fh2\u003E\u003Cp\u003E对于一个应用而言,除了需要开发HTML、CSS、JavaScript,通常还好使用到图片,字体文件,配置文件等诸多资源,那么前端工程化流程也就必然需要对这些资源进行处理与优化,最典型的说处理图片和字体文件。\u003C\u002Fp\u003E\u003Cp\u003E在Grunt或Gulp中,我们对图片和字体文件的处理通常是将其从源目录压缩优化处理后输出至输出目录,通常是以文件目录整体进行处理,每次构建时,对所有资源,包括未使用的图片均进行处理,效率是有局限的;而webpack中一切资源文件都可以处理成模块,然后在编译时管理模块依赖,可以做到只处理存在依赖的资源(即,使用了的资源)。\u003C\u002Fp\u003E\u003Ch4\u003E图片与字体\u003C\u002Fh4\u003E\u003Cp\u003E当我们在Js模块中引入CSS文件时,其中样式规则中的背景图片,字体文件如何处理呢?webpack只能管理模块化的东西,需要将其模块化,然后使用webpack管理依赖,webpack提供了file-loader加载器:\u003C\u002Fp\u003EFile Loader\u003Cblockquote\u003E\u003Cp\u003EInstructs webpack to emit the required object as file and to return its public url.\u003Cbr\u003E通知webpack将引入的对象输出为文件并返回其公开资源路径。\u003C\u002Fp\u003E\u003C\u002Fblockquote\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003Emodule: {\n
loaders: [\n
test: \u002F\\.(png|svg|jpe?g|gif)$\u002F,\n
loader: [\n
'file-loader'\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cul\u003E\u003Cli\u003E说明\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E当我们在js文件中import Image from '..\u002Fimages\u002Ftest.png'或在CSS文件中url('..\u002Fimages\u002Ftest.png')时,file-loader将处理该图片并在output.path目录下输出文件,然后将..\u002Fimages\u002Ftest.png路径替换成该输出文件路径。\u003C\u002Fp\u003E\u003Cp\u003E\u003Cem\u003E注,对于html中引用的图片,需要使用[html-loader]加载器处理(\u003Ca href=\&http:\u002F\\u002F?target=http%3A\u002F\u002Fnpm.taobao.org\u002Fpackage\u002Fhtml-loader\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003EPackage - html-loader\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E)。\u003C\u002Fem\u003E\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E\u003Cp\u003E参数\u003C\u002Fp\u003E\u003Col\u003E\u003Cli\u003EemitFile: 是否输出文件;\u003C\u002Fli\u003E\u003Cli\u003Ename: 指定输出文件的文件名,有几个可用内置变量:\u003Col\u003E\u003Cli\u003E[name]: 引用资源的名称;\u003C\u002Fli\u003E\u003Cli\u003E[path]: 引用资源的相对路径;\u003C\u002Fli\u003E\u003Cli\u003E[ext]: 资源拓展名;\u003C\u002Fli\u003E\u003Cli\u003E[hash]: 资源内容的hash值,默认使用md5算法计算得到,可以指定长度值,如[hash:7]表示返回hash值前7个字符;\u003C\u002Fli\u003E\u003Cli\u003E[hashType:hash:digestType:length]: 指定hash值计算算法类型和摘要类型,及摘要长度,如sha512:hash:base64:7表示使用sha512加密算法计算hash值并且返回7个字符的base64编码字符\u003C\u002Fli\u003E\u003C\u002Fol\u003E\u003C\u002Fli\u003E\u003C\u002Fol\u003E\u003C\u002Fli\u003E\u003Cli\u003E\u003Cp\u003E实例\u003C\u002Fp\u003E\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E在配置时可以指定参数:file-loader?name=[name].[ext]?[hash:8]或者以配置对象方式:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
test: \u002F\\.(png|svg|jpe?g|gif)$\u002F,\n
loaders: [\n
\u002F\u002F 'file-loader?name=[path][name].[ext]?[hash:8]'\n
loader: 'file-loader',\n
query: {\n
name: '[path][name].[ext]?[hash:8]'\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E对于CSS源代码:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E.wrapper {\n
font-size: 18\n
background: url('..\u002Fimages\u002Ftest.png') no-repeat 0 0;\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E输出CSS代码如下:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E.wrapper {\n
font-size: 18\n
background: url(assets\u002Fimages\u002Ftest.png?) no-repeat 0 0;\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003Eassets为output.publicPath指定值,images\u002Ftest.png?为配置文件中指定的name模板,在output.path目录下输出images\u002Ftest.png,区别是,不会携带?后的参数。\u003C\u002Fp\u003E\u003Cp\u003E另外,你也可以在js模板中这样使用:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
&img src={imgSrc} \u002F&\n\n
import imgSrc from 'path\u002Fxxx.png';\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003EUrl Loader\u003Cp\u003E你可能会发现前面并没有安装file-loader,因为有更好用的加载器url-loader,url-loader加载器是file-loader的升级版,他们唯一的不同之处在于:\u003C\u002Fp\u003E\u003Cp\u003Eurl-loader可以通过limit参数指定一个尺寸值,加载器会对小于该值的资源处理返回一个Data URL,以base64的方式嵌入HTML或CSS,如url-loader?limit=65000;对于大于该尺寸的资源将使用file-loader处理并且传递所有参数。\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003Emimetype\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E还可以设置mimetype对处理文件进行过滤,如url-loader?mimetype=image\u002Fpng将只处理png类型的资源。\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev url-loader\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cp\u003E该加载器对于图片和字体文件资源都适用:\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-js\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
\u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Etest\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&sr\&\u003E\u002F\\.(png|svg|jpe?g|gif)$\u002F\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Eloaders\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E[\u003C\u002Fspan\u003E\n
\u003Cspan class=\&c1\&\u003E\u002F\u002F 'url-loader?name=[path][name].[ext]?[hash:8]'\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Eloader\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E'url-loader'\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Equery\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Elimit\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&mi\&\u003EC\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Ename\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E'[path][name].[ext]?[hash:8]'\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E}\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E}\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E]\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E},\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Etest\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&sr\&\u003E\u002F\\.(woff|woff2|eot|ttf|otf)$\u002F\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Eloaders\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E[{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Eloader\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E'url-loader'\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Equery\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Elimit\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&mi\&\u003E1C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Ename\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E'[path][name].[ext]?[hash:8]'\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E}\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E}]\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E}\u003C\u002Fspan\u003E\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Ch4\u003E资源优化\u003C\u002Fh4\u003E\u003Cp\u003E完成以上配置后,已经可以在项目中很方便的引用各自资源了,但是通常我们还需要对图片字体等文件进行压缩优化处理,如Grunt中使用的imagemin插件一样压缩资源,webpack则提供了相关加载器img-loader。\u003C\u002Fp\u003E\u003Cul\u003E\u003Cli\u003E安装\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
npm install --save-dev img-loader\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cul\u003E\u003Cli\u003E配置\u003C\u002Fli\u003E\u003C\u002Ful\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-text\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E
test: \u002F\\.(jpe?g|png|gif|svg)$\u002Fi,\n
loaders: [\n
'url-loader?name=[path][name].[ext]?[hash:8]',\n
loader: 'img-loader',\n
options: {\n
\u002F\u002F 根据环境判断是否启用资源压缩\n
enabled: process.env.NODE_ENV === 'production', \n
gifsicle: {\n
interlaced: false \u002F\u002F 替换使用渐进式渲染\n
mozjpeg: {\n
progressive: true, \u002F\u002F 创建基准jpeg文件\n
optipng: {\n
optimizationLevel: 4, \u002F\u002F 优化级别,0-7,值越大,压缩越多\n
pngquant: {\n
quality: '75-90', \u002F\u002F 压缩质量,0-100,值越高,质量越高\n
speed: 3 \u002F\u002F 执行速度,0-10,速度过高质量受损,不建议过高\n
plugins: [\n
{ removeTitle: true }, \u002F\u002F 去除标题信息\n
{ convertPathData: false } \u002F\u002F 转换路径为更短的相对或决定路径\n
}\n\u003C\u002Fcode\u003E\u003C\u002Fpre\u003E\u003C\u002Fdiv\u003E\u003Cp\u003E以上为常见使用配置,更多详细配置信息请查看对应说明\u003Ca href=\&http:\u002F\\u002F?target=https%3A\\u002Fimagemin\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003Eimagemin文档\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E,特别注意的是上面使用了process.env.NODE_ENV当前环境变量,只有在生产环境启用图片压缩,因为压缩过程比较比较耗时,可能会降低开发、调试效率。\u003C\u002Fp\u003E\u003Ch4\u003E数据资源\u003C\u002Fh4\u003E\u003Cp\u003E对于数据类型文件资源,webpack内置支持加载解析.json文件,而其他类型则需要安装配置相应加载器,如.xml文件,需要安装并配置xml-loader。\u003C\u002Fp\u003E\u003Ch2\u003E资源管理的思考\u003C\u002Fh2\u003E\u003Cp\u003E在传统或稍早一点的应用中,我们通常会将所有的图片,字体等资源放在一个基础目录下,如assets\u002F或images,但是对于那些在多项目间重复的插件代码或资源来说,每一次迁移,我们都得在一大堆图片,字体资源里寻找出我们需要迁移的资源,这对代码可重用和其独立性有一定限制,而且与现在提倡的组件化开发模式也不相符。\u003C\u002Fp\u003E\u003Cp\u003Ewebpack对于资源的处理方式给组件化开发提供了很大便利,使得我们以组件为单位,可以在某一组件目录下存放所有相关的js,css,图片,字体等资源文件;组件的迁移公用成本很低。不过组件化开发并不是说不需要资源目录了,一些公用的资源依然放在项目的基础目录下。\u003C\u002Fp\u003E\u003Ch2\u003E说明\u003C\u002Fh2\u003E\u003Cp\u003E终于可以松口气,对于webpack管理CSS、图片、字体、数据资源的实践基本总结完成,其实感觉要介绍的还有很多,但是要尽量保证文章思路清晰,语句流畅,而且篇幅不能太长,水平有限,花费较多时间经历,希望能对读者有所帮助,后续篇章也会继续穿插介绍,力争本系列能较完整、较清晰地描述如何使用webpack开发SPA应用。\u003C\u002Fp\u003E\u003Cbr\u003E\u003Cp\u003E\u003Cstrong\u003E原创文章,转载请注明:\u003C\u002Fstrong\u003E 转载自 \u003Ca href=\&http:\u002F\\u002F?target=http%3A\u002F\\u002F\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003E熊建刚的博客\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E\u003C\u002Fp\u003E\u003Cp\u003E\u003Cstrong\u003E本文链接地址: \u003C\u002Fstrong\u003E\u003Ca href=\&http:\u002F\\u002F?target=http%3A\u002F\\u002F\u002F06\u002Fwebpack_resource_manage\u002F\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003Ewebpack与SPA实践之管理CSS等资源\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E\u003C\u002Fp\u003E&,&updated&:new Date(&T03:53:41.000Z&),&canComment&:false,&commentPermission&:&anyone&,&commentCount&:11,&collapsedCount&:0,&likeCount&:18,&state&:&published&,&isLiked&:false,&slug&:&&,&lastestTipjarors&:[],&isTitleImageFullScreen&:false,&rating&:&none&,&titleImage&:&&,&links&:{&comments&:&\u002Fapi\u002Fposts\u002F2Fcomments&},&reviewers&:[],&topics&:[{&url&:&https:\u002F\\u002Ftopic\u002F&,&id&:&&,&name&:&webpack&},{&url&:&https:\u002F\\u002Ftopic\u002F&,&id&:&&,&name&:&SPA (Single-Page App)&},{&url&:&https:\u002F\\u002Ftopic\u002F&,&id&:&&,&name&:&CSS&}],&adminClosedComment&:false,&titleImageSize&:{&width&:0,&height&:0},&href&:&\u002Fapi\u002Fposts\u002F&,&excerptTitle&:&&,&column&:{&slug&:&dreawer&,&name&:&极乐科技&},&tipjarState&:&activated&,&tipjarTagLine&:&真诚赞赏,手留余香&,&sourceUrl&:&&,&pageCommentsCount&:11,&tipjarorCount&:0,&annotationAction&:[],&hasPublishingDraft&:false,&snapshotUrl&:&&,&publishedTime&:&T11:53:41+08:00&,&url&:&\u002Fp\u002F&,&lastestLikers&:[{&bio&:&如有异议,你是对的。&,&isFollowing&:false,&hash&:&b0effbcc44410bbdfe07e50a&,&uid&:08,&isOrg&:false,&slug&:&shandian&,&isFollowed&:false,&description&:&VR虚拟现实从业者&,&name&:&王又贝&,&profileUrl&:&https:\u002F\\u002Fpeople\u002Fshandian&,&avatar&:{&id&:&c&,&template&:&https:\u002F\\u002F50\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false},{&bio&:&digger&,&isFollowing&:false,&hash&:&1df610efbc2b6efc74f26&,&uid&:24,&isOrg&:false,&slug&:&dmccer&,&isFollowed&:false,&description&:&&,&name&:&dmccer&,&profileUrl&:&https:\u002F\\u002Fpeople\u002Fdmccer&,&avatar&:{&id&:&da8e974dc&,&template&:&https:\u002F\\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false},{&bio&:&一个优秀的互联网宝宝&,&isFollowing&:false,&hash&:&14a2c53dc95df7e74da62e&,&uid&:496000,&isOrg&:false,&slug&:&mu-mu-zi-75-21&,&isFollowed&:false,&description&:&&,&name&:&木木子&,&profileUrl&:&https:\u002F\\u002Fpeople\u002Fmu-mu-zi-75-21&,&avatar&:{&id&:&v2-fd5c1a03c52&,&template&:&https:\u002F\\u002F50\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false},{&bio&:&&,&isFollowing&:false,&hash&:&1af7acfcaeb&,&uid&:20,&isOrg&:false,&slug&:&zhou-xiao-long-27&,&isFollowed&:false,&description&:&应用设计人员&,&name&:&赤虎&,&profileUrl&:&https:\u002F\\u002Fpeople\u002Fzhou-xiao-long-27&,&avatar&:{&id&:&v2-decd174201&,&template&:&https:\u002F\\u002F50\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false},{&bio&:&长安布衣&,&isFollowing&:false,&hash&:&d6fe9ff63b5f54e3d9365c4&,&uid&:68,&isOrg&:false,&slug&:&li-shu-88&,&isFollowed&:false,&description&:&卸甲一书生&,&name&:&梨树&,&profileUrl&:&https:\u002F\\u002Fpeople\u002Fli-shu-88&,&avatar&:{&id&:&v2-00d8e3af0b77f2f432f8d9c09c167868&,&template&:&https:\u002F\\u002F50\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false}],&summary&:&\u003Ca href=\&http:\u002F\\u002F\u002F31\u002Fwebpack_spa_build_env\u002F\& data-editable=\&true\& data-title=\&上一篇\& class=\&\&\u003E上一篇\u003C\u002Fa\u003E介绍了如何使用webpack搭建一个稳定的支持本地服务、自动刷新、模块热替换、使用ES6编写JavaScript的开发环境,本篇主要介绍webpack如何处理HTML应用三大元素的另一元素 – CSS及其他诸如图片、字体文件或者数据配置文件等资源。前言 在学习使用webpa…&,&reviewingCommentsCount&:0,&meta&:{&previous&:{&isTitleImageFullScreen&:false,&rating&:&none&,&titleImage&:&&,&links&:{&comments&:&\u002Fapi\u002Fposts\u002F2Fcomments&},&topics&:[{&url&:&https:\u002F\\u002Ftopic\u002F&,&id&:&&,&name&:&HTML&},{&url&:&https:\u002F\\u002Ftopic\u002F&,&id&:&&,&name&:&JavaScript&}],&adminClosedComment&:false,&href&:&\u002Fapi\u002Fposts\u002F&,&excerptTitle&:&&,&author&:{&bio&:&极乐小程序商店(http:\u002F\\u002F)&,&isFollowing&:false,&hash&:&4b43cf6f3d97f480b3ca2c13e5b12752&,&uid&:563500,&isOrg&:false,&slug&:&Dreawer&,&isFollowed&:false,&description&:&知乎专栏 && https:\u002F\\u002Fdreawer \n极乐小程序商店 && http:\u002F\\u002F\n极乐科技
&& http:\u002F\\u002F&,&name&:&极乐君&,&profileUrl&:&https:\u002F\\u002Fpeople\u002FDreawer&,&avatar&:{&id&:&v2-8cb80f97b7d5f74c7a8997cbc4089e1f&,&template&:&https:\u002F\\u002F50\u002F{id}_{size}.jpg&},&isOrgWhiteList&:false,&isBanned&:false},&column&:{&slug&:&dreawer&,&name&:&极乐科技&},&content&:&\u003Cp\u003E\u003Cb\u003E前言\u003C\u002Fb\u003E\u003C\u002Fp\u003E\u003Cp\u003E上一篇
\u003Ca class=\& wrap external\& href=\&http:\u002F\\u002F?target=http%3A\\u002Flandeanfen\u002Fp\u002F6776152.html\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003EJS组件系列——自己动手封装bootstrap-treegrid组件\u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E 博主自己动手封装了下treegrid的功能,但毕竟那个组件只是一个单独针对树形表格做的,适用性还比较有限。关注博主的小伙伴应该知道,博主的博客里面写了很多bootstrapTable的扩展,今天打算在直接在bootstrapTable的基础上扩展一个treegrid的功能,很多长期关注博主博客的小伙伴一直在问我怎么在bootstrapTable里面直接使用treegrid的功能,所以今天还是带来点福利。有兴趣的可以捧个人场!\u003C\u002Fp\u003E\u003Cp\u003E本文原创地址:\u003Ca href=\&http:\u002F\\u002F?target=http%3A\\u002Flandeanfen\u002Fp\u002F6924895.html\& class=\& wrap external\& target=\&_blank\& rel=\&nofollow noreferrer\&\u003EJS组件系列--自己动手扩展BootstrapTable的trefgrid功能 \u003Ci class=\&icon-external\&\u003E\u003C\u002Fi\u003E\u003C\u002Fa\u003E\u003C\u002Fp\u003E\u003Cbr\u003E\u003Ch1\u003E\u003Cb\u003E一、效果预览\u003C\u002Fb\u003E\u003C\u002Fh1\u003E\u003Cp\u003E全部折叠\u003Cbr\u003E\u003C\u002Fp\u003E\u003Cp\u003E\u003Cimg src=\&http:\u002F\\u002Fv2-fd409cc0b48aaa7ecc504fe2_b.png\& data-rawwidth=\&1646\& data-rawheight=\&371\& class=\&origin_image zh-lightbox-thumb\& width=\&1646\& data-original=\&http:\u002F\\u002Fv2-fd409cc0b48aaa7ecc504fe2_r.png\&\u003E展开一级\u003Cbr\u003E\u003C\u002Fp\u003E\u003Cp\u003E\u003Cimg src=\&http:\u002F\\u002Fv2-2ed90cf41f09a7de5cee5e_b.png\& data-rawwidth=\&1629\& data-rawheight=\&433\& class=\&origin_image zh-lightbox-thumb\& width=\&1629\& data-original=\&http:\u002F\\u002Fv2-2ed90cf41f09a7de5cee5e_r.png\&\u003E全部展开\u003Cbr\u003E\u003C\u002Fp\u003E\u003Cimg src=\&http:\u002F\\u002Fv2-6f7ee4d9cd52d1b6dcaa354_b.png\& data-rawwidth=\&1629\& data-rawheight=\&503\& class=\&origin_image zh-lightbox-thumb\& width=\&1629\& data-original=\&http:\u002F\\u002Fv2-6f7ee4d9cd52d1b6dcaa354_r.png\&\u003E\u003Cbr\u003E\u003Cbr\u003E\u003Ch1\u003E\u003Cb\u003E二、代码示例\u003C\u002Fb\u003E\u003C\u002Fh1\u003E\u003Cp\u003E怎么样?效果还行吧。给出js的源码供大家参考。\u003Cbr\u003E\u003C\u002Fp\u003E\u003Cdiv class=\&highlight\&\u003E\u003Cpre\u003E\u003Ccode class=\&language-js\&\u003E\u003Cspan\u003E\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E(\u003C\u002Fspan\u003E\u003Cspan class=\&kd\&\u003Efunction\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E(\u003C\u002Fspan\u003E\u003Cspan class=\&nx\&\u003E$\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E)\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&s1\&\u003E'use strict'\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E;\u003C\u002Fspan\u003E\n\n
\u003Cspan class=\&kd\&\u003Evar\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Esprintf\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&kd\&\u003Efunction\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E(\u003C\u002Fspan\u003E\u003Cspan class=\&nx\&\u003Estr\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E)\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&kd\&\u003Evar\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Eargs\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Earguments\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Eflag\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&kc\&\u003Etrue\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Ei\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&mi\&\u003E1\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E;\u003C\u002Fspan\u003E\n\n
\u003Cspan class=\&nx\&\u003Estr\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Estr\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E.\u003C\u002Fspan\u003E\u003Cspan class=\&nx\&\u003Ereplace\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E(\u003C\u002Fspan\u003E\u003Cspan class=\&sr\&\u003E\u002F%s\u002Fg\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E \u003Cspan class=\&kd\&\u003Efunction\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E()\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&kd\&\u003Evar\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Earg\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Eargs\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E[\u003C\u002Fspan\u003E\u003Cspan class=\&nx\&\u003Ei\u003C\u002Fspan\u003E\u003Cspan class=\&o\&\u003E++\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E];\u003C\u002Fspan\u003E\n\n
\u003Cspan class=\&k\&\u003Eif\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E(\u003C\u002Fspan\u003E\u003Cspan class=\&k\&\u003Etypeof\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Earg\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E===\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E'undefined'\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E)\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&nx\&\u003Eflag\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&kc\&\u003Efalse\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E;\u003C\u002Fspan\u003E\n
\u003Cspan class=\&k\&\u003Ereturn\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E''\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E;\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E}\u003C\u002Fspan\u003E\n
\u003Cspan class=\&k\&\u003Ereturn\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Earg\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E;\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E});\u003C\u002Fspan\u003E\n
\u003Cspan class=\&k\&\u003Ereturn\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Eflag\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E?\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Estr\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E:\u003C\u002Fspan\u003E \u003Cspan class=\&s1\&\u003E''\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E;\u003C\u002Fspan\u003E\n
\u003Cspan class=\&p\&\u003E};\u003C\u002Fspan\u003E\n\n
\u003Cspan class=\&kd\&\u003Evar\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003EgetFieldIndex\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=\u003C\u002Fspan\u003E \u003Cspan class=\&kd\&\u003Efunction\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E(\u003C\u002Fspan\u003E\u003Cspan class=\&nx\&\u003Ecolumns\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E,\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Efield\u003C\u002Fspan\u003E\u003Cspan class=\&p\&\u003E)\u003C\u002Fspan\u003E \u003Cspan class=\&p\&\u003E{\u003C\u002Fspan\u003E\n
\u003Cspan class=\&kd\&\u003Evar\u003C\u002Fspan\u003E \u003Cspan class=\&nx\&\u003Eindex\u003C\u002Fspan\u003E \u003Cspan class=\&o\&\u003E=

我要回帖

更多关于 webpack 只压缩css js 的文章

 

随机推荐