webpack+vue2.0+nodeJs搭建环境

webpack+vue2.0+nodeJs搭建环境

前言

写这篇技术文章的之前,花了一个礼拜的时间一直研究搭建环境,记录下所遇到的坑以及解决办法,这个环境在开发环境和生产环境下的使用,都记录了下来。

这个开发环境除了打包前端代码,然后也可以自动生成HTML文件,自动引入打包后的文件,开发过程中,如果改变了html、css、js文件,不用清除浏览器缓存,也不用刷新页面,浏览器自动更新。有没有很方便啊。哈哈,我搭建完环境开发的时候,简直无比方便。

搭建这个环境的时候,仿照一个大神的博客开始构建,但是里面遇到非常多的坑,自己一步一步的来解决,但是这个博客里用的webpack是1.0的。大家有兴趣也可以看看这个博客https://www.cnblogs.com/linfangshuhellowored/p/5657285.html

npm 和 nodejs还有淘宝镜像

npm 的全称是nodejs包管理,现在越来越多的项目(包)都可以通过npm来安装管理,nodejs是js运行在服务器端的平台,它使得js的能力进一步提高,我们还要使用nodejs配合 webpack 来完成热加载的功能。所以读者最好有nodejs的开发经验,如果有express的经验更好。

先安装node.js,然后在npm中install淘宝镜像

下载淘宝镜像的命令行

npm install -g cnpm--registry=https://registry.npm.taobao.org

让我们一步一步从零搭建这个项目。

首先新建一个目录,名为 myProject ,这是我们的项目目录。然后执行一些基本的步骤,比如 npm init 命令,在我们的项目中生成 package.json 文件,这几乎是必选的,因为我们的项目要有很多依赖,都是通过npm来管理的,而npm对于我们项目的管理,则是通过package.json文件:

执行npm init之后,会提示你填写一些项目的信息,一直回车默认就好了,或者直接执行 npm init -y 直接跳过询问步骤

然后我们新建一个叫做 app 的目录,这个是我们页面模块的目录,再在app目录下建立一个index目录,假设这个是首页模块的目录,然后再在index目录下建立一个 index.html 文件和 index.js 文件,分别是首页入口html文件和主js文件,然后再在index目录下建立一个components目录,这个目录用作存放首页组件模块的目录,因为我们最终要实现组件化开发。这样,当你完成上面的步骤后,你的项目看上去应该是这样的:

接下来通过npm安装项目依赖项(因为我安装了淘宝镜像,所以install的时候,用的是cnpm,没有安装淘宝镜像的可以用npm来下载依赖包):

cnpm install webpack webpack-dev-server vue-loader vue-html-loader css-loader vue-style-loader vue-hot-reload-api babel-loader babel-core babel-plugin-transform-runtime babel-preset-es2015 babel-runtime@5 --save-dev

下载完应该是这样的:

这个时候,你的package.json文件看起来应该是这样的:

我们安装了 babel 一系列包,用来解析ES6语法,因为我们使用ES6来开发项目,如果你不了解ES6语法,建议你看一看阮老师的教程,然后我们安装了一些loader包,比如css-loader/vue-loader等等,因为webpack是使用这些指定的loader去加载指定的文件的。

下来安装vue

cnpm install vue --save-dev

另外我们还使用 npm install vue –save 命令安装了 vue ,这个就是我们要在项目中使用的vue.js,我们可以直接像开发nodejs应用一样,直接require(‘vue’);即可,而不需要通过script标签引入,这一点在开发中很爽。

安装完了依赖,编辑以下文件并保存到相应位置:

1、index.html文件:

首页

2、index.js文件

import Vue from 'Vue'

import Favlist from './components/Favlist.vue'

Vue.config.debug = true;//开启错误提示

window.onload = function () {

new Vue({

el: '#app',

components: {

'my-component': Favlist

}

});

}

3、在components目录下新建一个 Favlist.vue 文件,作为我们的第一个组件:

div

要看懂上面的代码,你需要了解vue.js,假如你看不懂也没关系,我们首先在index.html中使用了自定义标签(即组件),然后在index.js中引入了Vue和我们的Favlist.vue组件,Favlist.vue文件中,我们使用了基本的vue组件语法,最后,我们希望它运行起来,这个时候,我们就需要webpack了。

// nodejs 中的path模块

var path = require('path');

module.exports = {

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: path.resolve(__dirname, '../app/index/index.js'),

// 输出配置

output: {

// 输出路径是 myProject/output/static

path: path.resolve(__dirname, '../output/static'),

publicPath: 'static/',

filename: '[name].[hash].js',

chunkFilename: '[id].[chunkhash].js'

},

resolve: {

extensions: ['*', '.js', '.vue']

},

// 特别注意:webpack v1 和webpack v2 的区别

module: {

rules: [

/* 用来解析vue后缀的文件 */

{

test: /\.vue$/,

loader: 'vue-loader'

},

/* 用babel来解析js文件并把es6的语法转换成浏览器认识的语法 */

{

test: /\.js$/,

loader: 'babel-loader',

/* 排除模块安装目录的文件 */

exclude: /node_modules/

},

{

test: /\.css$/,

use: [

{ loader: 'style-loader' },

{

loader: 'css-loader',

options: {

modules: true

}

},

]

}

]

}

}

上例中,相信你已经看懂了我的配置,入口文件是index.js文件,配置了相应输出,然后使用 vue-loader 去加载 .vue 结尾的文件,接下来我们就可以构建项目了,我们可以在命令行中执行:

错误提示我们应该选择合适的loader去加载这个 ‘./app/index/index.js’ 这个文件,并且说不期望index.js文件中的标识符(Unexpected token),这是因为我们使用了ES6的语法 import 语句,所以我们要使用 babel-loader 去加载我们的js文件,在配置文件中添加一个loaders项目,如下:

// nodejs 中的path模块

var path = require('path');

module.exports = {

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: path.resolve(__dirname, '../app/index/index.js'),

// 输出配置

output: {

// 输出路径是 myProject/output/static

path: path.resolve(__dirname, '../output/static'),

publicPath: 'static/',

filename: '[name].[hash].js',

chunkFilename: '[id].[chunkhash].js'

},

resolve: {

extensions: ['*', '.js', '.vue']

},

// 特别注意:webpack v1 和webpack v2 的区别

module: {

rules: [

/* 用来解析vue后缀的文件 */

{

test: /\.vue$/,

loader: 'vue-loader'

},

/* 用babel来解析js文件并把es6的语法转换成浏览器认识的语法 */

{

test: /\.js$/,

loader: 'babel-loader',

/* 排除模块安装目录的文件 */

exclude: /node_modules/

},

{

test: /\.css$/,

use: [

{ loader: 'style-loader' },

{

loader: 'css-loader',

options: {

modules: true

}

},

]

}

]

}

}

可是运行还是会得到如下错误提示:

这个错误提示说的是can't resolve 'babel',然后你需要访问https://webpack.js.org/guides/migrating/#automatic-loader-module-name-extension-removed这个网站

-loader引用装载机时不再可能省略扩展:

于是更改了loader

// nodejs 中的path模块

var path = require('path');

module.exports = {

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: path.resolve(__dirname, '../app/index/index.js'),

// 输出配置

output: {

// 输出路径是 myProject/output/static

path: path.resolve(__dirname, '../output/static'),

publicPath: 'static/',

filename: '[name].[hash].js',

chunkFilename: '[id].[chunkhash].js'

},

resolve: {

extensions: ['*', '.js', '.vue']

},

// 特别注意:webpack v1 和webpack v2 的区别

module: {

rules: [

/* 用来解析vue后缀的文件 */

{

test: /\.vue$/,

loader: 'vue-loader'

},

/* 用babel来解析js文件并把es6的语法转换成浏览器认识的语法 */

{

test: /\.js$/,

loader: 'babel-loader',

/* 排除模块安装目录的文件 */

exclude: /node_modules/

}

]

}

}

然后运行一下webpack

Cannot find module 'vue-template-compiler'

所以就需要安装一下这个'vue-template-compiler';

cnpm install vue-template-compiler --save-dev

然后再运行一下webpack。

此时出现了问题,我把项目文件里的node_modules删除,重新下载这些依赖包,重新下载依赖包,直接再命令行里npm install 就行了!这一步很重要

npm install

再运行webpack,

vue

但是虽然编译成功了,但是 vue的坑又来了

再index.html页面上引入打包后的js

令人百思不得其解的是,如果不使用 webpack 打包,而是直接在 HTML 文件中使用 script 标签引入 Favlist.js,在 JavaScript 中手写 MyComponent 的组件选项模板,完成的页面却可以正确显示,为什么这里会提示“模板或渲染函数未定义”呢?

这其实与 Vue 的两种不同的构建有关,在上述使用 webpack 打包的项目需要使用独立构建的 Vue 库,而在 node_modules/vue/package.json 文件中,已经通过 main 属性指定了通过 import Vue from ‘vue’ 或 require(‘vue’) 所引入的文件是 dist/vue.runtime.common.js,即运行时构建的 Vue 库。直接在 HTML 文件中使用 script 标签引入的是独立构建的 Vue 库,因此没有问题。

在包含单文件组件的项目中,使用 webpack 打包时已经将单文件组件中的模板预先编译成了渲染函数,因此一般情况下使用运行时构建的 Vue 库就可以了,但如果在使用 new Vue 创建 Vue 的根实例时,模板是从 el 挂载元素提取的,则需要使用独立构建的 Vue 库。

在使用 script 标签引入 Vue.js 的项目中,任意实例选项或组件选项中包含了 template 模板属性或从 el 挂载元素提取的模板时,均需要使用独立构建的 Vue 库。

要解决本文最开始的问题,需要在 webpack 配置中的 resolve 属性对象中添加如下 alias 设置:

resolve: {

extensions: ['*', '.js', '.vue'],

alias: {

'vue$': 'vue/dist/vue.common.js'

}

},

这里的 vue$ 表示精确匹配,由于 index.js 中还有一处大小写错误 import Vue from ‘Vue’,因此需要将 from 后面的 ‘Vue’ 修改为小写的 ‘vue’ 之后页面才能正确显示。

// nodejs 中的path模块

var path = require('path');

module.exports = {

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: path.resolve(__dirname, '../app/index/index.js'),

// 输出配置

output: {

// 输出路径是 myProject/output/static

path: path.resolve(__dirname, '../output/static'),

publicPath: 'static/',

filename: '[name].[hash].js',

chunkFilename: '[id].[chunkhash].js'

},

resolve: {

extensions: ['*', '.js', '.vue'],

alias: {

'vue$': 'vue/dist/vue.common.js'

}

},

// 特别注意:webpack v1 和webpack v2 的区别

module: {

rules: [

/* 用来解析vue后缀的文件 */

{

test: /\.vue$/,

loader: 'vue-loader'

},

/* 用babel来解析js文件并把es6的语法转换成浏览器认识的语法 */

{

test: /\.js$/,

loader: 'babel-loader',

/* 排除模块安装目录的文件 */

exclude: /node_modules/

},

{

test: /\.css$/,

use: [

{ loader: 'style-loader' },

{

loader: 'css-loader',

options: {

modules: true

}

},

]

}

]

}

}

index.js是这样的,把import Vue from 'Vue' from 后面的 ‘Vue’ 修改为小写的 ‘vue’ 之后页面才能正确显示。

import Vue from 'vue'

import Favlist from './components/Favlist'

Vue.config.debug = true;//开启错误提示

window.onload = function () {

new Vue({

el: '#app',

components: {

'my-component': Favlist

}

});

}

重新运行一下webpack,然后重新再index中引入打包后的js,运行页面,就会

自动化构建HTML文件。

然后用浏览器打开这个页面,你可以看到你写的代码正确的执行了。

首先安装 html-webpack-plugin

cnpm install html-webpack-plugin --save-dev

然后修改配置项

// nodejs 中的path模块

var path = require('path');

var HtmlWebpackPlugin = require('html-webpack-plugin')

module.exports = {

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: path.resolve(__dirname, '../app/index/index.js'),

// 输出配置

output: {

// 输出路径是 myProject/output/static

path: path.resolve(__dirname, '../output/static'),

publicPath: 'static/',

filename: '[name].[hash].js',

chunkFilename: '[id].[chunkhash].js'

},

resolve: {

extensions: ['*', '.js', '.vue'],

alias: {

'vue$': 'vue/dist/vue.common.js'

}

},

// 特别注意:webpack v1 和webpack v2 的区别

module: {

rules: [

/* 用来解析vue后缀的文件 */

{

test: /\.vue$/,

loader: 'vue-loader'

},

/* 用babel来解析js文件并把es6的语法转换成浏览器认识的语法 */

{

test: /\.js$/,

loader: 'babel-loader',

/* 排除模块安装目录的文件 */

exclude: /node_modules/

},

{

test: /\.css$/,

use: [

{ loader: 'style-loader' },

{

loader: 'css-loader',

options: {

modules: true

}

},

]

}

]

},

plugins: [

new HtmlWebpackPlugin({

filename: '../index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

]

}

webpack热加载

然后再次执行构建命令,成功之后,看你的输出目录,多出来一个index.html文件,双击它,代码正确执行,你可以打开这个文件查看一下,webpack自动帮我们引入了相应的文件。

问题继续来了,难道每次我们都要构建之后才能查看运行的代码吗?那岂不是很没有效率,别担心,webpack提供了几种方式,进行热加载,在开发模式中,我们使用这种方式来提高效率,这里要介绍的,是使用webpack-dev-middleware中间件和webpack-hot-middleware中间件,首先安装两个中间件:

cnpm install webpack-dev-middleware webpack-hot-middleware --save-dev

另外,还要安装express,这是一个nodejs框架

cnpm install express --save-dev

先简单介绍一下这俩个插件

我们之前所面临的问题是,如果我们的代码改动了,我们要想看到浏览器的变化,需要先对项目进行构建,然后才能查看效果,这样对于开发效率来讲,简直就是不可忍受的一件事,试想我仅仅修改一个背景颜色就要构建一下项目,这肯定太繁琐了,好在有webpack-dev-middleware中间件,它是对webpack一个简单的包装,它可以通过连接服务器服务那些从webpack发射出来的文件,它有一下几点好处:

1、不会向硬盘写文件,而是在内存中,注意我们构建项目实际就是向硬盘写文件。

2、当文件改变的时候,这个中间件不会再服务旧的包,你可以直接刷新浏览器就能看到最新的效果,这样你就不必等待构建的时间,所见即所得。

下面我们在build目录中创建一个 dev-server.js 的文件,并写入一下内容:

// 引入必要的模块

var express = require('express')

var webpack = require('webpack')

var config = require('./webpack.config')

// 创建一个express实例

var app = express()

// 调用webpack并把配置传递过去

var compiler = webpack(config)

// 使用 webpack-dev-middleware 中间件

var devMiddleware = require('webpack-dev-middleware')(compiler, {

publicPath: config.output.publicPath,

stats: {

colors: true,

chunks: false

}

})

// 注册中间件

app.use(devMiddleware)

// 监听 8888端口,开启服务器

app.listen(8888, function (err) {

if (err) {

console.log(err)

return

}

console.log('Listening at http://localhost:8888')

})

然后运行启动一下nodeJs

node build/dev-server.js

如果看到下图所示,证明你的服务成功开启了:

接下来打开浏览器,输入:

http://localhost:8888/app/index/index.html

如果不出意外,你会得到一个404,如下图:

// 输出配置

output: {

// 输出路径是 myProject/output/static

path: path.resolve(__dirname, '../output/static'),

publicPath: '/',

filename: '[name].[hash].js',

chunkFilename: '[id].[chunkhash].js'

},

2、将 plugins 中 HtmlWebpackPlugin 中的 filename 修改为 ‘app/index/index.html’

plugins: [

new HtmlWebpackPlugin({

filename: 'app/index/index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

]

重启服务,再刷新页面,如果看到如下界面,证明你成功了:

node build/dev-server.js

var HtmlWebpackPlugin = require('html-webpack-plugin')

var path = require('path');

// 引入基本配置

var config = require('./webpack.config');

config.plugins = [

new HtmlWebpackPlugin({

filename: 'app/index/index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

module.exports = config;

这样,我们在dev环境下的配置文件中覆盖了基本配置文件,我们只需要在dev-server.js中将

var config = require('./webpack.config')

修改为:

然后,重启服务,刷新浏览器,你应该得到同样的成功结果,而这一次当我们执行构建命令:

现在我们已经使用 webpack-dev-middleware 搭建基本的开发环境了,但是我们并不满足,因为我们每次都要手动去刷新浏览器,所谓的热加载,意思就是说能够追踪我们代码的变化,并自动更新界面,甚至还能保留程序状态。要完成热加载,我们就需要使用另外一个中间件webpack-hot-middleware

2 、webpack-hot-middleware

webpack-hot-middleware只配合 webpack-dev-middleware 使用,它能给你提供热加载。

它的使用很简单,总共分4步:

1、安装,我们上面已经安装过了

var HtmlWebpackPlugin = require('html-webpack-plugin')

var path = require('path');

var webpack = require('webpack');

// 引入基本配置

var config = require('./webpack.config');

config.plugins = [

// 添加三个插件

new webpack.optimize.OccurenceOrderPlugin(),

new webpack.HotModuleReplacementPlugin(),

new webpack.NoErrorsPlugin(),

new HtmlWebpackPlugin({

filename: 'app/index/index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

module.exports = config;

entry: ['webpack-hot-middleware/client', path.resolve(__dirname, '../app/index/index.js')],

4、在 dev-server.js 文件中使用插件

// 引入必要的模块

var express = require('express')

var webpack = require('webpack')

var config = require('./webpack.dev.conf')

// 创建一个express实例

var app = express()

// 调用webpack并把配置传递过去

var compiler = webpack(config)

// 使用 webpack-dev-middleware 中间件

var devMiddleware = require('webpack-dev-middleware')(compiler, {

publicPath: config.output.publicPath,

stats: {

colors: true,

chunks: false

}

})

// 使用 webpack-hot-middleware 中间件

var hotMiddleware = require('webpack-hot-middleware')(compiler)

// 注册中间件

app.use(devMiddleware)

// 注册中间件

app.use(hotMiddleware)

// 监听 8888端口,开启服务器

app.listen(8888, function (err) {

if (err) {

console.log(err)

return

}

console.log('Listening at http://localhost:8888')

})

但是启动服务的时候会报错

使用webpack命令的时候报错,Webpack.optimize.OccurenceOrderPlugin is not a constructor

解决方法:OccurenceOrderPlugin更改为OccurrenceOrderPlugin 即可

var HtmlWebpackPlugin = require('html-webpack-plugin')

var path = require('path');

var webpack = require('webpack');

// 引入基本配置

var config = require('./webpack.config');

config.plugins = [

// 添加三个插件

new webpack.optimize.OccurrenceOrderPlugin(),

new webpack.HotModuleReplacementPlugin(),

new webpack.NoErrorsPlugin(),

new HtmlWebpackPlugin({

filename: 'app/index/index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

module.exports = config;

ok,现在重启的服务,然后修改 Favlist.vue 中的color为 ‘green’,不用启动服务也不用刷新页面,直接就可以看到效果

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: ['webpack-hot-middleware/client', path.resolve(__dirname, '../app/index/index.js')],

var HtmlWebpackPlugin = require('html-webpack-plugin')

var path = require('path');

var webpack = require('webpack');

// 引入基本配置

var config = require('./webpack.config');

config.plugins = [

new webpack.optimize.OccurrenceOrderPlugin(),

new webpack.HotModuleReplacementPlugin(),

new webpack.NoErrorsPlugin(),

new HtmlWebpackPlugin({

filename: 'app/index/index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

// 动态向入口配置中注入 webpack-hot-middleware/client

var devClient = 'webpack-hot-middleware/client';

Object.keys(config.entry).forEach(function (name, i) {

var extras = [devClient]

config.entry[name] = extras.concat(config.entry[name])

})

module.exports = config;

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: {

index: path.resolve(__dirname, '../app/index/index.js')

},

再次查看浏览器,发现可以热加载。但是这样就结束了吗?还没有,修改了 index.html 文件,看看会不会热加载,实际上不会,你还是需要手动刷新页面,为了能够当 index.html 文件的改动也能够触发自动刷新,我们还需要做一些工作。

第一步:在 dev-server.js 文件中监听html文件改变事件,修改后的 dev-server.js 文件如下:

// 引入必要的模块

var express = require('express')

var webpack = require('webpack')

var config = require('./webpack.dev.conf')

// 创建一个express实例

var app = express()

// 调用webpack并把配置传递过去

var compiler = webpack(config)

// 使用 webpack-dev-middleware 中间件

var devMiddleware = require('webpack-dev-middleware')(compiler, {

publicPath: config.output.publicPath,

stats: {

colors: true,

chunks: false

}

})

var hotMiddleware = require('webpack-hot-middleware')(compiler)

// webpack插件,监听html文件改变事件

compiler.plugin('compilation', function (compilation) {

compilation.plugin('html-webpack-plugin-after-emit', function (data, cb) {

// 发布事件

hotMiddleware.publish({ action: 'reload' })

cb()

})

})

// 注册中间件

app.use(devMiddleware)

// 注册中间件

app.use(hotMiddleware)

// 监听 8888端口,开启服务器

app.listen(8888, function (err) {

if (err) {

console.log(err)

return

}

console.log('Listening at http://localhost:8888')

})

从上面的代码上看,我们增加了如下的代码

// webpack插件,监听html文件改变事件

compiler.plugin('compilation', function (compilation) {

compilation.plugin('html-webpack-plugin-after-emit', function (data, cb) {

// 发布事件

hotMiddleware.publish({ action: 'reload' })

cb()

})

})

这段代码可能看不懂。涉及到webpack的插件编写,可以参考github上的代码

https://github.com/webpack/docs/wiki/plugins

https://github.com/webpack/docs/wiki/How-to-write-a-plugin

var HtmlWebpackPlugin = require('html-webpack-plugin')

var path = require('path');

var webpack = require('webpack');

// 引入基本配置

var config = require('./webpack.config');

config.plugins = [

new webpack.optimize.OccurrenceOrderPlugin(),

new webpack.HotModuleReplacementPlugin(),

new webpack.NoErrorsPlugin(),

new HtmlWebpackPlugin({

filename: 'app/index/index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

// var devClient = 'webpack-hot-middleware/client';

var devClient = './build/dev-client';

Object.keys(config.entry).forEach(function (name, i) {

var extras = [devClient]

config.entry[name] = extras.concat(config.entry[name])

})

module.exports = config;

我们修改了devClient变量,将 ‘webpack-hot-middleware/client’ 替换成 ‘./build/dev-client’,最终会导致,我们入口配置会变成下面这样:

entry: {

index: [

'./build/dev-client',

path.resolve(__dirname, '../app/index/index.js')

]

},

第三步:新建 build/dev-client.js 文件,并编辑如下内容:

var hotClient = require('webpack-hot-middleware/client')

// 订阅事件,当 event.action === 'reload' 时执行页面刷新

hotClient.subscribe(function (event) {

if (event.action === 'reload') {

window.location.reload()

}

})

现在重新启动服务,然后去改变app/index/index.html文件的时候,页面就会自动刷新

开发环境终于搞定了,下面我们再来谈一谈生产环境,也就是构建输出,我们现在可以执行一下构建命令,看看输出的内容是什么,为了不必每次都要输入下面这条长命令:

我们在 package.json 文件中添加 “scripts” 项,如下图:

"scripts": {

// "test": "echo \"Error: no test specified\" && exit 1"

"build":"webpack --display-modules --display-chunks --config build/webpack.config.js",

"dev":"node ./build/dev-server.js"

},

"scripts": {// "test": "echo \"Error: no test specified\" && exit 1" "build":"webpack --display-modules --display-chunks --config build/webpack.config.js", "dev":"node ./build/dev-server.js" },

这样,我们就可以通过执行下面命令来进行构建,同时我们还增加了一条开启开发服务器的命令:

// 构建

npm run build

// 开启开发服务器

npm run dev

现在我们只有一个js文件输出了,并没有css文件输出,在生产环境,我们希望css文件生成单独的文件,所以我们要使用 extract-text-webpack-plugin 插件,安装:

cnpm install extract-text-webpack-plugin --save-dev

var HtmlWebpackPlugin = require('html-webpack-plugin')

var ExtractTextPlugin = require('extract-text-webpack-plugin')

var path = require('path');

var webpack = require('webpack');

// 引入基本配置

var config = require('./webpack.config');

config.vue = {

loaders: {

css: ExtractTextPlugin.extract("css")

}

};

config.plugins = [

// 提取css为单文件

new ExtractTextPlugin("../[name].[contenthash].css"),

new HtmlWebpackPlugin({

filename: '../index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

module.exports = config;

然后修改 package.json 文件中的 script 项为如下:

"scripts": {

"build": "webpack --display-modules --display-chunks --config build/webpack.prod.conf.js",

"dev": "node ./build/dev-server.js"

},

config.plugins = [

// 提取css为单文件

new ExtractTextPlugin("../[name].[contenthash].css"),

new HtmlWebpackPlugin({

filename: '../index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

}),

new webpack.DefinePlugin({

'process.env': {

NODE_ENV: '"production"'

}

}),

// 压缩代码

new webpack.optimize.UglifyJsPlugin({

compress: {

warnings: false

}

}),

new webpack.optimize.OccurenceOrderPlugin(),

// 提取css为单文件

new ExtractTextPlugin("../[name].[contenthash].css"),

new HtmlWebpackPlugin({

filename: '../index.html',

template: path.resolve(__dirname, '../app/index/index.html'),

inject: true

})

];

new webpack.optimize.CommonsChunkPlugin({

name: 'vendors',

filename: 'vendors.js',

}),

// 入口文件,path.resolve()方法,可以结合我们给定的两个参数最后生成绝对路径,最终指向的就是我们的index.js文件

entry: {

index: path.resolve(__dirname, '../app/index/index.js'),

vendors: [

'Vue'

]

},

上面代码的意思是,我们把Vue.js当做公共模块单独打包,你可以在这个数组中增加其他模块,一起作为公共模块打包成一个文件,我们执行构建命令,然后查看输出,如下图,成功提取:

即使在清贫的岁月,也不能失去对幸福美好的向往,那些摆脱平庸的梦总能编制我们简单的生活,为我们简单的时光点缀希望。不能说我们总要多热爱生活,但总要有一颗懂得欣赏和珍惜的心。 from《布鲁克林有棵树》

  • 发表于:
  • 原文链接http://kuaibao.qq.com/s/20180203G06SGB00?refer=cp_1026
  • 腾讯「云+社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。

扫码关注云+社区

领取腾讯云代金券