Stories

Detail Return Return

vue+vuecli+webapck2項目配置文件詳解 - Stories Detail

1.文件結構

├─build
│   ├─build.js
│   ├─check-versions.js
│   ├─dev-client.js
│   ├─dev-server.js
│   ├─utils.js
│   ├─vue-loader.conf.js
│   ├─webpack.base.conf.js
│   ├─webpack.dev.conf.js
│   ├─webpack.prod.conf.js
│   └─webpack.test.conf.js
├─config
│   ├─dev.env.js
│   ├─index.js
│   ├─prod.env.js
│   └─test.env.js
├─...
└─package.json

2.package.json文件

package.json裏面的scripts字段:

"scripts": {
    "dev": "node build/dev-server.js",
    "build": "node build/build.js",
    "unit": "cross-env BABEL_ENV=test karma start test/unit/karma.conf.js --single-run",
    "e2e": "node test/e2e/runner.js",
    "test": "npm run unit && npm run e2e",
    "lint": "eslint --ext .js,.vue src test/unit/specs test/e2e/specs"
  }

運行”npm run dev”:執行build/dev-server.js文件
運行”npm run build”:執行build/build.js文件

3.build文件夾分析

build/dev-server.js

該文件主要作用:

  • 檢查node和npm的版本、引入相關插件和配置
  • webpack對源碼進行編譯打包並返回compiler對象
  • 創建express服務器
  • 配置開發中間件(webpack-dev-middleware)和熱重載中間件(webpack-hot-middleware)
  • 掛載代理服務和中間件
  • 配置靜態資源
  • 啓動服務器監聽特定端口(8080)
  • 自動打開瀏覽器並打開特定網址(localhost:8080)
説明: express服務器提供靜態文件服務,不過它還使用了http-proxy-middleware,一個http請求代理的中間件。前端開發過程中需要使用到後台的API的話,可以通過配置proxyTable來將相應的後台請求代理到專用的API服務器。

代碼詳細註釋:

// 檢查NodeJS和npm的版本
require('./check-versions')()

// 獲取基本配置
var config = require('../config')
// 如果Node的環境變量中沒有設置當前的環境(NODE_ENV),則使用config中的dev環境配置作為當前的環境
if (!process.env.NODE_ENV) {
  process.env.NODE_ENV = JSON.parse(config.dev.env.NODE_ENV)
}

// opn是一個可以調用默認軟件打開網址、圖片、文件等內容的插件
// 這裏用它來調用默認瀏覽器打開dev-server監聽的端口,例如:localhost:8080
var opn = require('opn')
var path = require('path')
var express = require('express')
var webpack = require('webpack')
// http-proxy-middleware是一個express中間件,用於將http請求代理到其他服務器
// 例:localhost:8080/api/xxx  -->  localhost:3000/api/xxx
// 這裏使用該插件可以將前端開發中涉及到的請求代理到提供服務的後台服務器上,方便與服務器對接
var proxyMiddleware = require('http-proxy-middleware')
// 開發環境下的webpack配置
var webpackConfig = require('./webpack.dev.conf')

// dev-server 監聽的端口,如果沒有在命令行傳入端口號,則使用config.dev.port設置的端口,例如8080
var port = process.env.PORT || config.dev.port
// 用於判斷是否要自動打開瀏覽器的布爾變量,當配置文件中沒有設置自動打開瀏覽器的時候其值為 false
var autoOpenBrowser = !!config.dev.autoOpenBrowser
// HTTP代理表,指定規則,將某些API請求代理到相應的服務器
var proxyTable = config.dev.proxyTable
// 創建express服務器
var app = express()
// webpack根據配置開始編譯打包源碼並返回compiler對象
var compiler = webpack(webpackConfig)
// webpack-dev-middleware將webpack編譯打包後得到的產品文件存放在內存中而沒有寫進磁盤
// 將這個中間件掛到express上使用之後即可提供這些編譯後的產品文件服務
var devMiddleware = require('webpack-dev-middleware')(compiler, {
  publicPath: webpackConfig.output.publicPath, // 設置訪問路徑為webpack配置中的output裏面所對應的路徑
  quiet: true // 設置為true,使其不要在控制枱輸出日誌
})
// webpack-hot-middleware,用於實現熱重載功能的中間件
var hotMiddleware = require('webpack-hot-middleware')(compiler, {
  log: false, // 關閉控制枱的日誌輸出
  heartbeat: 2000 // 發送心跳包的頻率
})
// webpack(重新)編譯打包完成後並將js、css等文件inject到html文件之後,通過熱重載中間件強制頁面刷新
compiler.plugin('compilation', function (compilation) {
  compilation.plugin('html-webpack-plugin-after-emit', function (data, cb) {
    hotMiddleware.publish({ action: 'reload' })
    cb()
  })
})

// 根據 proxyTable 中的代理請求配置來設置express服務器的http代理規則
Object.keys(proxyTable).forEach(function (context) {
  var options = proxyTable[context]
  // 格式化options,例如將'www.example.com'變成{ target: 'www.example.com' }
  if (typeof options === 'string') {
    options = { target: options }
  }
  app.use(proxyMiddleware(options.filter || context, options))
})

// handle fallback for HTML5 history API
// 重定向不存在的URL,用於支持SPA(單頁應用)
// 例如使用vue-router並開啓了history模式
app.use(require('connect-history-api-fallback')())

// serve webpack bundle output
// 掛載webpack-dev-middleware中間件,提供webpack編譯打包後的產品文件服務
app.use(devMiddleware)

// enable hot-reload and state-preserving
// compilation error display
// 掛載熱重載中間件
app.use(hotMiddleware)

// serve pure static assets
// 提供static文件夾上的靜態文件服務
var staticPath = path.posix.join(config.dev.assetsPublicPath, config.dev.assetsSubDirectory)
app.use(staticPath, express.static('./static'))

// 訪問鏈接
var uri = 'http://localhost:' + port

// 創建promise,在應用服務啓動之後resolve
// 便於外部文件require了這個dev-server之後的代碼編寫
var _resolve
var readyPromise = new Promise(resolve => {
  _resolve = resolve
})

console.log('> Starting dev server...')
// webpack-dev-middleware等待webpack完成所有編譯打包之後輸出提示語到控制枱,表明服務正式啓動
// 服務正式啓動才自動打開瀏覽器進入頁面
devMiddleware.waitUntilValid(() => {
  console.log('> Listening at ' + uri + '\n')
  // when env is testing, don't need open it
  if (autoOpenBrowser && process.env.NODE_ENV !== 'testing') {
    opn(uri)
  }
  _resolve()
})

// 啓動express服務器並監聽相應的端口
var server = app.listen(port)

// 暴露本模塊的功能給外部使用,例如下面這種用法
// var devServer = require('./build/dev-server')
// devServer.ready.then(() => {...})
// if (...) { devServer.close() }
module.exports = {
  ready: readyPromise,
  close: () => {
    server.close()
  }
}

build/webpack.base.conf.js

主要作用:

  • 配置webpack編譯入口
  • 配置webpack輸出路徑和命名規則
  • 配置模塊resolve規則
  • 配置不同類型模塊的處理規則
説明: 這個配置裏面只配置了.js、.vue、圖片、字體等幾類文件的處理規則,如果需要處理其他文件可以在module.rules裏面另行配置。

代碼詳細註釋:

var path = require('path')
var fs = require('fs')
var utils = require('./utils')
var config = require('../config')
var vueLoaderConfig = require('./vue-loader.conf')

// 獲取絕對路徑
function resolve (dir) {
  return path.join(__dirname, '..', dir)
}

module.exports = {
  // webpack入口文件
  entry: {
    app: './src/main.js'
  },
  // webpack輸出路徑和命名規則
  output: {
    // webpack輸出的目標文件夾路徑(例如:/dist)
    path: config.build.assetsRoot,
    // webpack輸出bundle文件命名格式
    filename: '[name].js',
    // webpack編譯輸出的發佈路徑(例如'//cdn.xxx.com/app/')
    publicPath: process.env.NODE_ENV === 'production'
      ? config.build.assetsPublicPath
      : config.dev.assetsPublicPath
  },
  // 模塊resolve的規則
  resolve: {
    extensions: ['.js', '.vue', '.json'],
    // 別名,方便引用模塊,例如有了別名之後,
    // import Vue from 'vue/dist/vue.common.js'可以寫成 import Vue from 'vue'
    alias: {
      'vue$': 'vue/dist/vue.esm.js',
      '@': resolve('src'),
    },
    symlinks: false
  },
  // 不同類型模塊的處理規則
  module: {
    rules: [
      {// 對src和test文件夾下的.js和.vue文件使用eslint-loader進行代碼規範檢查
        test: /\.(js|vue)$/,
        loader: 'eslint-loader',
        enforce: 'pre',
        include: [resolve('src'), resolve('test')],
        options: {
          formatter: require('eslint-friendly-formatter')
        }
      },
      {// 對所有.vue文件使用vue-loader進行編譯
        test: /\.vue$/,
        loader: 'vue-loader',
        options: vueLoaderConfig
      },
      {// 對src和test文件夾下的.js文件使用babel-loader將es6+的代碼轉成es5
        test: /\.js$/,
        loader: 'babel-loader',
        include: [resolve('src'), resolve('test')]
      },
      {// 對圖片資源文件使用url-loader
        test: /\.(png|jpe?g|gif|svg)(\?.*)?$/,
        loader: 'url-loader',
        options: {
          // 小於10K的圖片轉成base64編碼的dataURL字符串寫到代碼中
          limit: 10000,
          // 其他的圖片轉移到靜態資源文件夾
          name: utils.assetsPath('img/[name].[hash:7].[ext]')
        }
      },
      {// 對多媒體資源文件使用url-loader
        test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/,
        loader: 'url-loader',
        options: {
          // 小於10K的資源轉成base64編碼的dataURL字符串寫到代碼中
          limit: 10000,
          // 其他的資源轉移到靜態資源文件夾
          name: utils.assetsPath('media/[name].[hash:7].[ext]')
        }
      },
      {// 對字體資源文件使用url-loader
        test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/,
        loader: 'url-loader',
        options: {
          // 小於10K的資源轉成base64編碼的dataURL字符串寫到代碼中
          limit: 10000,
          // 其他的資源轉移到靜態資源文件夾
          name: utils.assetsPath('fonts/[name].[hash:7].[ext]')
        }
      }
    ]
  }
}

build/webpack.dev.conf.js

主要作用:

  • 將webpack的熱重載客户端代碼添加到每個entry對應的應用
  • 合併基礎的webpack配置
  • 配置樣式文件的處理規則,styleLoaders
  • 配置Source Maps
  • 配置webpack插件

詳細代碼註釋:

var utils = require('./utils')
var webpack = require('webpack')
var config = require('../config')
// webpack-merge是一個可以合併數組和對象的插件
var merge = require('webpack-merge')
var baseWebpackConfig = require('./webpack.base.conf')
// html-webpack-plugin用於將webpack編譯打包後的產品文件注入到html模板中
// 即自動在index.html裏面加上<link>和<script>標籤引用webpack打包後的文件
var HtmlWebpackPlugin = require('html-webpack-plugin')
// friendly-errors-webpack-plugin用於更友好地輸出webpack的警告、錯誤等信息
var FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin')

// add hot-reload related code to entry chunks
// 給每個入口頁面(應用)加上dev-client,用於跟dev-server的熱重載插件通信,實現熱更新
Object.keys(baseWebpackConfig.entry).forEach(function (name) {
  baseWebpackConfig.entry[name] = ['./build/dev-client'].concat(baseWebpackConfig.entry[name])
})

module.exports = merge(baseWebpackConfig, {
  module: {
    // 樣式文件的處理規則,對css/sass/scss等不同內容使用相應的styleLoaders
    // 由utils配置出各種類型的預處理語言所需要使用的loader,例如sass需要使用sass-loader
    rules: utils.styleLoaders({ sourceMap: config.dev.cssSourceMap })
  },
  // cheap-module-eval-source-map is faster for development
  // 使用這種source-map更快
  devtool: '#cheap-module-eval-source-map',
  // webpack插件
  plugins: [
    new webpack.DefinePlugin({
      'process.env': config.dev.env
    }),
    // 開啓webpack熱更新功能
    new webpack.HotModuleReplacementPlugin(),
    // webpack編譯過程中出錯的時候跳過報錯階段,不會阻塞編譯,在編譯結束後報錯
    new webpack.NoEmitOnErrorsPlugin(),
    // 自動將依賴注入html模板,並輸出最終的html文件到目標文件夾
    new HtmlWebpackPlugin({
      filename: 'index.html',
      template: 'index.html',
      inject: true
    }),
    new FriendlyErrorsPlugin()
  ]
})

build/webpack.prod.conf.js

主要作用:

  • 合併基礎的webpack配置
  • 配置樣式文件的處理規則,styleLoaders
  • 配置webpack的輸出
  • 配置webpack插件
  • gzip模式下的webpack插件配置
  • webpack-bundle分析
説明: webpack插件裏面多了醜化壓縮代碼以及抽離css文件等插件。

詳細代碼註釋:

var path = require('path')
var utils = require('./utils')
var webpack = require('webpack')
var config = require('../config')
var merge = require('webpack-merge')
var baseWebpackConfig = require('./webpack.base.conf')
/* copy-webpack-plugin,用於將static中的靜態文件複製到產品文件夾dist */
var CopyWebpackPlugin = require('copy-webpack-plugin')
var HtmlWebpackPlugin = require('html-webpack-plugin')
var ExtractTextPlugin = require('extract-text-webpack-plugin')
/* optimize-css-assets-webpack-plugin,用於優化和最小化css資源  */
var OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin')

var env = config.build.env

var webpackConfig = merge(baseWebpackConfig, {
  module: {
    /* 樣式文件的處理規則,對css/sass/scss等不同內容使用相應的styleLoaders */
    /* 由utils配置出各種類型的預處理語言所需要使用的loader,例如sass需要使用sass-loader */
    rules: utils.styleLoaders({
      sourceMap: config.build.productionSourceMap,
      extract: true
    })
  },
  /* 是否使用source-map */
  devtool: config.build.productionSourceMap ? '#source-map' : false,
  /* webpack輸出路徑和命名規則 */
  output: {
    path: config.build.assetsRoot,
    filename: utils.assetsPath('js/[name].[chunkhash].js'),
    chunkFilename: utils.assetsPath('js/[id].[chunkhash].js')
  },
  /* webpack插件 */
  plugins: [
    /* http://vuejs.github.io/vue-loader/en/workflow/production.html */
    new webpack.DefinePlugin({
      'process.env': env
    }),
    /* 醜化壓縮JS代碼 */
    new webpack.optimize.UglifyJsPlugin({
      compress: {
        warnings: false
      },
      sourceMap: true
    }),
    /* extract css into its own file */
    /* 將css提取到單獨的文件 */
    new ExtractTextPlugin({
      filename: utils.assetsPath('css/[name].[contenthash].css')
    }),
    /* Compress extracted CSS. We are using this plugin so that possible */
    /* duplicated CSS from different components can be deduped. */
    /* 優化、最小化css代碼,如果只簡單使用extract-text-plugin可能會造成css重複 */
    /* 具體原因可以看npm上面optimize-css-assets-webpack-plugin的介紹 */
    new OptimizeCSSPlugin({
      cssProcessorOptions: {
        safe: true
      }
    }),
    /* generate dist index.html with correct asset hash for caching. */
    /* you can customize output by editing /index.html */
    /* see https://github.com/ampedandwired/html-webpack-plugin */
    /* 將產品文件的引用注入到index.html */
    new HtmlWebpackPlugin({
      filename: config.build.index,
      template: 'index.html',
      inject: true,
      minify: {
        /* 刪除index.html中的註釋 */
        removeComments: true,
        /* 刪除index.html中的空格 */
        collapseWhitespace: true,
        /* 刪除各種html標籤屬性值的雙引號 */
        removeAttributeQuotes: true
        /* more options: */
        /* https://github.com/kangax/html-minifier#options-quick-reference */
      },
      /* necessary to consistently work with multiple chunks via CommonsChunkPlugin */
      /* 注入依賴的時候按照依賴先後順序進行注入,比如,需要先注入vendor.js,再注入app.js */
      chunksSortMode: 'dependency'
    }),
    /* keep module.id stable when vender modules does not change */
    new webpack.HashedModuleIdsPlugin(),
    /* split vendor js into its own file */
    /* 將所有從node_modules中引入的js提取到vendor.js,即抽取庫文件 */
    new webpack.optimize.CommonsChunkPlugin({
      name: 'vendor',
      minChunks: function (module, count) {
        /* any required modules inside node_modules are extracted to vendor */
        return (
          module.resource &&
          /\.js$/.test(module.resource) &&
          module.resource.indexOf(
            path.join(__dirname, '../node_modules')
          ) === 0
        )
      }
    }),
    /* extract webpack runtime and module manifest to its own file in order to */
    /* prevent vendor hash from being updated whenever app bundle is updated */
    /* 從vendor中提取出manifest,原因如上 */
    new webpack.optimize.CommonsChunkPlugin({
      name: 'manifest',
      chunks: ['vendor']
    }),
    /* copy custom static assets */
    /* 將static文件夾裏面的靜態資源複製到dist/static */
    new CopyWebpackPlugin([
      {
        from: path.resolve(__dirname, '../static'),
        to: config.build.assetsSubDirectory,
        ignore: ['.*']
      }
    ])
  ]
})

/* 如果開啓了產品gzip壓縮,則利用插件將構建後的產品文件進行壓縮 */
if (config.build.productionGzip) {
  /* 一個用於壓縮的webpack插件 */
  var CompressionWebpackPlugin = require('compression-webpack-plugin')

  webpackConfig.plugins.push(
    new CompressionWebpackPlugin({
      asset: '[path].gz[query]',
      /* 壓縮算法 */
      algorithm: 'gzip',
      test: new RegExp(
        '\\.(' +
        config.build.productionGzipExtensions.join('|') +
        ')$'
      ),
      threshold: 10240,
      minRatio: 0.8
    })
  )
}

/* 如果啓動了report,則通過插件給出webpack構建打包後的產品文件分析報告 */
if (config.build.bundleAnalyzerReport) {
  var BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin
  webpackConfig.plugins.push(new BundleAnalyzerPlugin())
}

module.exports = webpackConfig

build/utils.js

utils提供工具函數,包括生成處理各種樣式語言的loader,獲取資源文件存放路徑的工具函數。

  • 計算資源文件存放路徑
  • 生成cssLoaders用於加載.vue文件中的樣式
  • 生成styleLoaders用於加載不在.vue文件中的單獨存在的樣式文件

詳細代碼註釋:

var path = require('path')
var config = require('../config')
// extract-text-webpack-plugin可以提取bundle中的特定文本,將提取後的文本單獨存放到另外的文件
// 這裏用來提取css樣式
var ExtractTextPlugin = require('extract-text-webpack-plugin')

// 資源文件的存放路徑
exports.assetsPath = function (_path) {
  var assetsSubDirectory = process.env.NODE_ENV === 'production'
    ? config.build.assetsSubDirectory
    : config.dev.assetsSubDirectory
  return path.posix.join(assetsSubDirectory, _path)
}

// 生成css、sass、scss等各種用來編寫樣式的語言所對應的loader配置
exports.cssLoaders = function (options) {
  options = options || {}
  // css-loader配置
  var cssLoader = {
    loader: 'css-loader',
    options: {
      // 是否最小化
      minimize: process.env.NODE_ENV === 'production',
      // 是否使用source-map
      sourceMap: options.sourceMap
    }
  }

  // generate loader string to be used with extract text plugin
  // 生成各種loader配置,並且配置了extract-text-pulgin
  function generateLoaders (loader, loaderOptions) {
    // 默認是css-loader
    var loaders = [cssLoader]
    // 如果非css,則增加一個處理預編譯語言的loader並設好相關配置屬性
    // 例如generateLoaders('less'),這裏就會push一個less-loader
    // less-loader先將less編譯成css,然後再由css-loader去處理css
    // 其他sass、scss等語言也是一樣的過程
    if (loader) {
      loaders.push({
        loader: loader + '-loader',
        options: Object.assign({}, loaderOptions, {
          sourceMap: options.sourceMap
        })
      })
    }

    // Extract CSS when that option is specified
    // (which is the case during production build)
    if (options.extract) {
      // 配置extract-text-plugin提取樣式
      return ExtractTextPlugin.extract({
        use: loaders,
        fallback: 'vue-style-loader'
      })
    } else {
      // 無需提取樣式則簡單使用vue-style-loader配合各種樣式loader去處理<style>裏面的樣式
      return ['vue-style-loader'].concat(loaders)
    }
  }

  // https://vue-loader.vuejs.org/en/configurations/extract-css.html
  // 得到各種不同處理樣式的語言所對應的loader
  return {
    css: generateLoaders(),
    postcss: generateLoaders(),
    less: generateLoaders('less'),
    sass: generateLoaders('sass', { indentedSyntax: true }),
    scss: generateLoaders('sass'),
    stylus: generateLoaders('stylus'),
    styl: generateLoaders('stylus')
  }
}

// Generate loaders for standalone style files (outside of .vue)
// 生成處理單獨的.css、.sass、.scss等樣式文件的規則
exports.styleLoaders = function (options) {
  var output = []
  var loaders = exports.cssLoaders(options)
  for (var extension in loaders) {
    var loader = loaders[extension]
    output.push({
      test: new RegExp('\\.' + extension + '$'),
      use: loader
    })
  }
  return output
}

build/vue-loader.conf.js

vue-loader.conf的配置比較簡單,詳細代碼註釋:

var utils = require('./utils')
var config = require('../config')
var isProduction = process.env.NODE_ENV === 'production'

module.exports = {
  // 處理.vue文件中的樣式
  loaders: utils.cssLoaders({
    // 是否打開source-map
    sourceMap: isProduction
      ? config.build.productionSourceMap
      : config.dev.cssSourceMap,
    // 是否提取樣式到單獨的文件
    extract: isProduction
  }),
  postcss: [
    require('autoprefixer')({
      browsers: ['iOS >= 7', 'Android >= 4.1']
    })
  ]
}

build/dev-client.js

dev-client.js裏面主要寫了瀏覽器端代碼,用於實現webpack的熱更新。

/* eslint-disable */
// 實現瀏覽器端的EventSource,用於跟服務器雙向通信
// webpack熱重載客户端跟dev-server上的熱重載插件之間需要進行雙向通信
// 服務端webpack重新編譯後,會向客户端推送信息,告訴客户端進行更新
require('eventsource-polyfill')
// webpack熱重載客户端
var hotClient = require('webpack-hot-middleware/client?noInfo=true&reload=true')

// 客户端收到更新動作,執行頁面刷新
hotClient.subscribe(function (event) {
  if (event.action === 'reload') {
    window.location.reload()
  }
})

build/build.js

執行”npm run build”的時候首先執行的是build/build.js文件,主要作用:

  • loading動畫
  • 刪除目標文件夾
  • 執行webpack構建
  • 輸出信息

詳細代碼解釋:

// 檢查NodeJS和npm的版本
require('./check-versions')()

process.env.NODE_ENV = 'production'

// ora,一個可以在終端顯示spinner的插件
var ora = require('ora')
// rm,用於刪除文件或文件夾的插件
var rm = require('rimraf')
var path = require('path')
// chalk,用於在控制枱輸出帶顏色字體的插件
var chalk = require('chalk')
var webpack = require('webpack')
var config = require('../config')
var webpackConfig = require('./webpack.prod.conf')

var spinner = ora('building for production...')
spinner.start() // 開啓loading動畫

// 首先將整個dist文件夾以及裏面的內容刪除,以免遺留舊的沒用的文件
// 刪除完成後才開始webpack構建打包
rm(path.join(config.build.assetsRoot, config.build.assetsSubDirectory), err => {
  if (err) throw err
  // 執行webpack構建打包,完成之後在終端輸出構建完成的相關信息或者輸出報錯信息並退出程序
  webpack(webpackConfig, function (err, stats) {
    spinner.stop()
    if (err) throw err
    process.stdout.write(stats.toString({
      colors: true,
      modules: false,
      children: false,
      chunks: false,
      chunkModules: false
    }) + '\n\n')

    if (stats.hasErrors()) {
      console.log(chalk.red('  Build failed with errors.\n'))
      process.exit(1)
    }

    console.log(chalk.cyan('  Build complete.\n'))
    console.log(chalk.yellow(
      '  Tip: built files are meant to be served over an HTTP server.\n' +
      '  Opening index.html over file:// won\'t work.\n'
    ))
  })
})

build/check-versions.js

check-version.js完成對node和npm的版本檢測,詳細代碼解釋:

// chalk, 用於在控制枱輸出帶顏色字體的插件
var chalk = require('chalk')
// semver, 語義化版本檢查插件(The semantic version parser used by npm)
var semver = require('semver')
var packageConfig = require('../package.json')
// shelljs, 執行Unix命令行的插件
var shell = require('shelljs')
// 開闢子進程執行指令cmd並返回結果
function exec (cmd) {
  return require('child_process').execSync(cmd).toString().trim()
}

// node和npm版本需求
var versionRequirements = [
  {
    name: 'node',
    currentVersion: semver.clean(process.version),
    versionRequirement: packageConfig.engines.node
  }
]

if (shell.which('npm')) {
  versionRequirements.push({
    name: 'npm',
    currentVersion: exec('npm --version'),
    versionRequirement: packageConfig.engines.npm
  })
}

module.exports = function () {
  var warnings = []
  // 依次判斷版本是否符合要求
  for (var i = 0; i < versionRequirements.length; i++) {
    var mod = versionRequirements[i]
    if (!semver.satisfies(mod.currentVersion, mod.versionRequirement)) {
      warnings.push(mod.name + ': ' +
        chalk.red(mod.currentVersion) + ' should be ' +
        chalk.green(mod.versionRequirement)
      )
    }
  }
  // 如果有警告則將其輸出到控制枱
  if (warnings.length) {
    console.log('')
    console.log(chalk.yellow('To use this template, you must update following to modules:'))
    console.log()
    for (var i = 0; i < warnings.length; i++) {
      var warning = warnings[i]
      console.log('  ' + warning)
    }
    console.log()
    process.exit(1)
  }
}

4.config文件夾分析

config/index.js

這裏面描述了開發和構建兩種環境下的配置,前面的build文件夾下也有不少文件引用了index.js裏面的配置。詳細代碼註釋如下:

// see http://vuejs-templates.github.io/webpack for documentation.
var path = require('path')

module.exports = {
  // 構建產品時使用的配置
  build: {
    // 環境變量
    env: require('./prod.env'),
    // html入口文件
    index: path.resolve(__dirname, '../dist/index.html'),
    // 產品文件的存放路徑
    assetsRoot: path.resolve(__dirname, '../dist'),
    // 二級目錄,存放靜態資源文件的目錄,位於dist文件夾下
    assetsSubDirectory: 'static',
    // 發佈路徑,如果構建後的產品文件有用於發佈CDN或者放到其他域名的服務器,可以在這裏進行設置
    // 設置之後構建的產品文件在注入到index.html中的時候就會帶上這裏的發佈路徑
    assetsPublicPath: '/',
    // 是否使用source-map
    productionSourceMap: true,
    // Gzip off by default as many popular static hosts such as
    // Surge or Netlify already gzip all static assets for you.
    // Before setting to `true`, make sure to:
    // npm install --save-dev compression-webpack-plugin
    // 是否開啓gzip壓縮
    productionGzip: false,
    // gzip模式下需要壓縮的文件的擴展名,設置js、css之後就只會對js和css文件進行壓縮
    productionGzipExtensions: ['js', 'css'],
    // Run the build command with an extra argument to
    // View the bundle analyzer report after build finishes:
    // `npm run build --report`
    // Set to `true` or `false` to always turn it on or off
    // 是否展示webpack構建打包之後的分析報告
    bundleAnalyzerReport: process.env.npm_config_report
  },
  // 開發過程中使用的配置
  dev: {
    // 環境變量
    env: require('./dev.env'),
    // dev-server監聽的端口
    port: 8080,
    // 是否自動打開瀏覽器
    autoOpenBrowser: true,
    // 靜態資源文件夾
    assetsSubDirectory: 'static',
    // 發佈路徑
    assetsPublicPath: '/',
    // 代理配置表,在這裏可以配置特定的請求代理到對應的API接口
    // 例如將'localhost:8080/api/xxx'代理到'www.example.com/api/xxx'
    proxyTable: {},
    // CSS Sourcemaps off by default because relative paths are "buggy"
    // with this option, according to the CSS-Loader README
    // (https://github.com/webpack/css-loader#sourcemaps)
    // In our experience, they generally work as expected,
    // just be aware of this issue when enabling this option.
    // 是否開啓 cssSourceMap
    cssSourceMap: false
  }
}

config/dev.env.js、config/prod.env.js和config/test.env.js

這三個文件就簡單設置了環境變量。

user avatar honwhy Avatar tosmile Avatar 6fafa Avatar populus Avatar wszgrcy Avatar aresn Avatar fushengruomengweihuanjihe Avatar songxianling1992 Avatar huaiyug Avatar nidexiaoxiongruantangna Avatar lpicker Avatar patelo Avatar
Favorites 33 users favorite the story!
Favorites

Add a new Comments

Some HTML is okay.