都8102年了,如今还来谈webpack的配置,额,是有点晚了。并且,基于vue-cli或者create-react-app生成的项目,也已经一键为咱们配置好了webpack,看起来彷佛并不须要咱们深刻了解。javascript
不过,为了学习和理解webpack解决了前端的哪些痛点,仍是有必要从零开始本身搭建一个简单的开发环境。本文的webpack配置参考了vue-cli提供webpack-simple
模板,这也是vue-cli里面最简单的一个webpack配置,很是适合从零开始学习。css
注: 本文webpack基于3.10.0html
npm i webpack -g
复制代码
新建一个文件夹vue-webpack-simple前端
新建package.jsonvue
npm init -y
复制代码
安装vue webpack webpack-dev-serverjava
npm i vue --save
复制代码
npm i webpack webpack-dev-server --save-dev
复制代码
根目录下新建index.htmlnode
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
</body>
</html>
复制代码
根目录下新建webpack.config.jsreact
var path = require('path');
var webpack = require('webpack');
module.exports = {};
复制代码
新建src文件夹,src文件夹下新建main.jswebpack
目前整个项目的结构以下: git
在ES6出现以前,js是没有统一的模块体系。 服务器端使用CommonJS规范,而浏览器端又有AMD和CMD两种规范
webpack的思想就是一切皆模块,官方推荐使用commonJS规范,这使得咱们浏览器端也可使用commonJS的模块化写法
module.exports = {}
复制代码
src目录下新建一个util.js
module.exports = function say() {
console.log('hello world');
}
复制代码
main.js
var say = require('./util');
say();
复制代码
修改webpack.config.js
var path = require('path');
var webpack = require('webpack');
module.exports = {
entry: './src/main.js', // 项目的入口文件,webpack会从main.js开始,把全部依赖的js都加载打包
output: {
path: path.resolve(__dirname, './dist'), // 项目的打包文件路径
publicPath: '/dist/', // 经过devServer访问路径
filename: 'build.js' // 打包后的文件名
},
devServer: {
historyApiFallback: true,
overlay: true
}
};
复制代码
修改package.josn
"scripts": {
"dev": "webpack-dev-server --open --hot",
"build": "webpack --progress --hide-modules"
},
复制代码
注意:webpack-dev-server会自动启动一个静态资源web服务器 --hot参数表示启动热更新
修改index.html,引入打包后的文件
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<script src="/dist/build.js"></script>
</body>
</html>
复制代码
运行
npm run dev
复制代码
能够发现浏览器自动打开的一个页面,查看控制台,有hello world
打出
咱们随意修改util.js,能够发现浏览器会自动刷新,很是方便。
若是咱们但愿看打包后的bundle.js文件,运行
npm run build
复制代码
能够看到生成了一个dist目录,里面就有打包好后的bundle.js
webpack默认不支持转码es6,可是import
export
这两个语法却单独支持。因此咱们能够改写前面的模块化写法
util.js
export default function say() {
console.log('hello world ');
}
复制代码
main.js
import say from './util';
say();
复制代码
下面咱们来试着引入vue(目前不考虑单文件.vue)
main.js
import Vue from 'vue';
var app = new Vue({
el: '#app',
data: {
message: 'Hello Vue!'
}
});
复制代码
index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<div id="app">
{{message}}
</div>
<script src="/dist/build.js"></script>
</body>
</html>
复制代码
还要注意一点:要修改webpack.config.js文件
var path = require('path');
var webpack = require('webpack');
module.exports = {
entry: './src/main.js',
output: {
path: path.resolve(__dirname, './dist'),
publicPath: '/dist/',
filename: 'build.js'
},
devServer: {
historyApiFallback: true,
overlay: true
},
resolve: {
alias: {
'vue$': 'vue/dist/vue.esm.js'
}
}
};
复制代码
从新运行npm run dev,能够看到,页面正常显示了Hello World
webpack默认只支持js的模块化,若是须要把其余文件也当成模块引入,就须要相对应的loader解析器
npm i node-sass css-loader vue-style-loader sass-loader --save-dev
复制代码
webpack.config.js
var path = require('path');
var webpack = require('webpack');
module.exports = {
entry: './src/main.js',
output: {
path: path.resolve(__dirname, './dist'),
publicPath: '/dist/',
filename: 'build.js'
},
devServer: {
historyApiFallback: true,
overlay: true
},
resolve: {
alias: {
'vue$': 'vue/dist/vue.esm.js'
}
},
module: {
rules: [
{
test: /\.css$/,
use: [
'vue-style-loader',
'css-loader'
],
}
]
}
};
复制代码
解释:
{
test: /\.css$/,
use: [
'vue-style-loader',
'css-loader'
],
}
复制代码
这段代码意思是:匹配后缀名为css的文件,而后分别用css-loader,vue-style-loader去解析 解析器的执行顺序是从下往上(先css-loader再vue-style-loader)
注意:由于咱们这里用vue开发,因此使用vue-style-loader,其余状况使用style-loader
css-loader使得咱们能够用模块化的写法引入css,vue-style-loader会将引入的css插入到html页面里的style标签里
要引入scss也是同理的配置写法:
module: {
rules: [
{
test: /\.css$/,
use: [
'vue-style-loader',
'css-loader'
],
},
{
test: /\.scss$/,
use: [
'vue-style-loader',
'css-loader',
'sass-loader'
],
},
{
test: /\.sass$/,
use: [
'vue-style-loader',
'css-loader',
'sass-loader?indentedSyntax'
],
}]
}
复制代码
咱们如今来试下 在src目录下新建style目录,style目录里新建common.scss
body {
background: #fed;
}
复制代码
main.js
import './style/common.scss';
复制代码
发现css样式有用了
ES6的语法大多数浏览器依旧不支持,bable能够把ES6转码成ES5语法,这样咱们就能够大胆的在项目中使用最新特性了
npm i babel-core babel-loader babel-preset-env babel-preset-stage-3 --save-dev
复制代码
在项目根目录新建一个.babelrc文件
{
"presets": [
["env", { "modules": false }],
"stage-3"
]
}
复制代码
webpack.config.js添加一个loader
{
test: /\.js$/,
loader: 'babel-loader',
exclude: /node_modules/
}
复制代码
exclude表示忽略node_modules文件夹下的文件,不用转码
如今咱们来试下async await语法吧 util.js
export default function getData() {
return new Promise((resolve, reject) => {
resolve('ok');
})
}
复制代码
main.js
import getData from './util';
import Vue from 'vue';
import './style/common.scss';
var app = new Vue({
el: '#app',
data: {
message: 'Hello Vue!'
},
methods: {
async fetchData() {
const data = await getData();
this.message = data;
}
},
created() {
this.fetchData();
}
});
复制代码
这时控制台会报一个错误regeneratorRuntime is not defined
,由于咱们没有安装babel-polyfill
npm i babel-polyfill --save-dev
复制代码
而后修改webpack.config.js的入口
entry: ['babel-polyfill', './src/main.js'],
复制代码
从新npm run dev,能够发现正常运行了
把图片也当成模块引入
npm i file-loader --save-dev
复制代码
webpack.config.js添加一个loader
{
test: /\.(png|jpg|gif|svg)$/,
loader: 'file-loader',
options: {
name: '[name].[ext]?[hash]'
}
}
复制代码
在src目录下新建一个img目录,存放一张图片logo.png
修改main.js
import getData from './util';
import Vue from 'vue';
import './style/common.scss';
Vue.component('my-component', {
template: '<img :src="url" />',
data() {
return {
url: require('./img/logo.png')
}
}
})
var app = new Vue({
el: '#app',
data: {
message: 'Hello Vue !'
},
methods: {
async fetchData() {
const data = await getData();
this.message = data;
}
},
created() {
this.fetchData()
}
});
复制代码
修改index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<div id="app">
{{message}}
<my-component/>
</div>
<script src="/dist/build.js"></script>
</body>
</html>
复制代码
能够看见,图片也被正确加载了
在前面的例子里,咱们使用 Vue.component 来定义全局组件 在实际项目里,更推荐使用单文件组件
npm i vue-loader vue-template-compiler --save-dev
复制代码
添加一个loader
{
test: /\.vue$/,
loader: 'vue-loader',
options: {
loaders: {
'scss': [
'vue-style-loader',
'css-loader',
'sass-loader'
],
'sass': [
'vue-style-loader',
'css-loader',
'sass-loader?indentedSyntax'
]
}
}
}
复制代码
在src目录下新建一个App.vue
<template>
<div id="app">
<h1>{{ msg }}</h1>
<img src="./img/logo.png">
<input type="text" v-model="msg">
</div>
</template>
<script> import getData from './util'; export default { name: 'app', data () { return { msg: 'Welcome to Your Vue.js' } }, created() { this.fetchData(); }, methods: { async fetchData() { const data = await getData(); this.msg = data; } } } </script>
<style lang="scss"> #app { font-family: "Avenir", Helvetica, Arial, sans-serif; h1 { color: green; } } </style>
复制代码
main.js
import Vue from 'vue';
import App from './App.vue';
import './style/common.scss';
new Vue({
el: '#app',
template: '<App/>',
components: { App }
})
复制代码
index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<div id="app"></div>
<script src="/dist/build.js"></script>
</body>
</html>
复制代码
npm run dev,能够发现单文件被正确加载了
在开发阶段,调试也是很是重要的一项需求。
App.vue
created() {
this.fetchData();
console.log('23333');
}
复制代码
咱们故意打一个console,打开控制台
咱们点击进入这个console的详细地址
进入的是打包后的build.js,我并不知道是在哪一个组件里写的,这就形成了调试困难
这时就要修改webpack.config.js
module.exports = {
entry: ['babel-polyfill', './src/main.js'],
// 省略其余...
devtool: '#eval-source-map'
};
复制代码
从新npm run dev
咱们先试着npm run build打包一下文件
会发现,打包后的build.js很是大,有500多k了
在实际发布时,会对文件进行压缩,缓存,分离等等优化处理
npm i cross-env --save-dev
复制代码
修改package.json
"scripts": {
"dev": "cross-env NODE_ENV=development webpack-dev-server --open --hot",
"build": "cross-env NODE_ENV=production webpack --progress --hide-modules"
}
复制代码
此次咱们设置了环境变量,打包时,NODE_ENV是production
而后修改webpack.config.js,判断NODE_ENV为production时,压缩js代码
var path = require('path');
var webpack = require('webpack');
module.exports = {
// 省略...
}
if (process.env.NODE_ENV === 'production') {
module.exports.devtool = '#source-map';
module.exports.plugins = (module.exports.plugins || []).concat([
new webpack.DefinePlugin({
'process.env': {
NODE_ENV: '"production"'
}
}),
new webpack.optimize.UglifyJsPlugin(),
])
}
复制代码
从新打包
能够看见,压缩效果很是明显!
至此,一个很是简单的vue开发环境搭建成功。
注意:本文中的配置还有很是多能够优化的地方,好比分离js和css
读者能够自行了解相关知识,这里只是带领你们了解最基础的webpack配置。