├── .babelrc ├── .editorconfig ├── .gitignore ├── .postcssrc.js ├── README.md ├── build ├── build.js ├── check-versions.js ├── logo.png ├── utils.js ├── vue-loader.conf.js ├── webpack.base.conf.js ├── webpack.dev.conf.js └── webpack.prod.conf.js ├── config ├── dev.env.js ├── index.js ├── multipage.js ├── multipage.project.js ├── prod.env.js └── test.env.js ├── dist ├── admin │ ├── favicon.ico │ ├── index.html │ └── static │ │ ├── css │ │ ├── index.3a947c1c1f207832148c696e78c1f1bd.css │ │ └── index.3a947c1c1f207832148c696e78c1f1bd.css.map │ │ └── js │ │ ├── index.3675151582d13b8e0c7e.js │ │ ├── index.3675151582d13b8e0c7e.js.map │ │ ├── manifest.2ae2e69a05c33dfc65f8.js │ │ ├── manifest.2ae2e69a05c33dfc65f8.js.map │ │ ├── vendor.7fed9fa7b7ba482410b7.js │ │ └── vendor.7fed9fa7b7ba482410b7.js.map └── master │ ├── favicon.ico │ ├── index.html │ └── static │ ├── css │ ├── index.7cb8a531774c9910737734e1c15db140.css │ └── index.7cb8a531774c9910737734e1c15db140.css.map │ └── js │ ├── index.0f86190d29b252e0535a.js │ ├── index.0f86190d29b252e0535a.js.map │ ├── manifest.2ae2e69a05c33dfc65f8.js │ ├── manifest.2ae2e69a05c33dfc65f8.js.map │ ├── vendor.7fed9fa7b7ba482410b7.js │ └── vendor.7fed9fa7b7ba482410b7.js.map ├── package.json ├── src ├── admin │ ├── components │ │ └── HelloWorld.vue │ ├── config │ │ └── index.js │ ├── pages │ │ └── index │ │ │ ├── App.vue │ │ │ ├── favicon.ico │ │ │ ├── index.html │ │ │ └── index.js │ └── router │ │ └── index.js ├── global │ └── assets │ │ └── logo.png └── master │ ├── components │ └── HelloWorld.vue │ ├── config │ └── index.js │ ├── pages │ └── index │ │ ├── App.vue │ │ ├── favicon.ico │ │ ├── index.html │ │ └── index.js │ └── router │ └── index.js ├── static └── .gitkeep └── test └── unit ├── .eslintrc ├── jest.conf.js ├── setup.js └── specs └── HelloWorld.spec.js /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | ["env", { 4 | "modules": false, 5 | "targets": { 6 | "browsers": ["> 1%", "last 2 versions", "not ie <= 8"] 7 | } 8 | }], 9 | "stage-2" 10 | ], 11 | "plugins": ["transform-vue-jsx", "transform-runtime"], 12 | "env": { 13 | "test": { 14 | "presets": ["env", "stage-2"], 15 | "plugins": ["transform-vue-jsx", "transform-es2015-modules-commonjs", "dynamic-import-node"] 16 | } 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | charset = utf-8 5 | indent_style = space 6 | indent_size = 2 7 | end_of_line = lf 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules/ 3 | /dist/ 4 | npm-debug.log* 5 | yarn-debug.log* 6 | yarn-error.log* 7 | /test/unit/coverage/ 8 | 9 | # Editor directories and files 10 | .idea 11 | .vscode 12 | *.suo 13 | *.ntvs* 14 | *.njsproj 15 | *.sln 16 | /.history/ 17 | package-lock.json -------------------------------------------------------------------------------- /.postcssrc.js: -------------------------------------------------------------------------------- 1 | // https://github.com/michael-ciniawsky/postcss-load-config 2 | 3 | module.exports = { 4 | "plugins": { 5 | "postcss-import": {}, 6 | "postcss-url": {}, 7 | // to edit target browsers: use "browserslist" field in package.json 8 | "autoprefixer": {} 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # vue-multi-project 2 | 布置一套vue+webpack,实现多个项目。 3 | 4 | ### 欢迎 Star ,您的Star是对我的鼓励! 5 | 6 | ### 前提 7 | 目前工作出现一个问题,多个项目均使用vue+webpack,并且项目之间有很多的组件或者方法都是相同的(粘贴复制),整体布局基本一样(只是主题颜色有些许不同)。感觉这样子后期维护繁杂,同一个问题需要修改很多遍。于是,在研究两天之后,生成了一套新的项目结构,基本改善了上述问题。 8 | 9 | ### 工程结构 10 | ``` 11 | The files tree is: 12 | ================= 13 | |__ build 14 | |__ build.js 15 | |__ check-versions.js 16 | |__ logo.png 17 | |__ utils.js 18 | |__ vue-loader.conf.js 19 | |__ webpack.base.conf.js 20 | |__ webpack.dev.conf.js 21 | |__ webpack.prod.conf.js 22 | |__ config 23 | |__ dev.env.js 24 | |__ index.js 25 | |__ multipage.js 26 | |__ multipage.project.js 27 | |__ prod.env.js 28 | |__ test.env.js 29 | |__ dist 30 | |__ admin 31 | |__ favicon.ico 32 | |__ index.html 33 | |__ static 34 | |__ css 35 | |__ index.3a947c1c1f207832148c696e78c1f1bd.css 36 | |__ index.3a947c1c1f207832148c696e78c1f1bd.css.map 37 | |__ js 38 | |__ index.3675151582d13b8e0c7e.js 39 | |__ index.3675151582d13b8e0c7e.js.map 40 | |__ manifest.2ae2e69a05c33dfc65f8.js 41 | |__ manifest.2ae2e69a05c33dfc65f8.js.map 42 | |__ vendor.7fed9fa7b7ba482410b7.js 43 | |__ vendor.7fed9fa7b7ba482410b7.js.map 44 | |__ master 45 | |__ favicon.ico 46 | |__ index.html 47 | |__ static 48 | |__ css 49 | |__ index.7cb8a531774c9910737734e1c15db140.css 50 | |__ index.7cb8a531774c9910737734e1c15db140.css.map 51 | |__ js 52 | |__ index.0f86190d29b252e0535a.js 53 | |__ index.0f86190d29b252e0535a.js.map 54 | |__ manifest.2ae2e69a05c33dfc65f8.js 55 | |__ manifest.2ae2e69a05c33dfc65f8.js.map 56 | |__ vendor.7fed9fa7b7ba482410b7.js 57 | |__ vendor.7fed9fa7b7ba482410b7.js.map 58 | |__ node_modules 59 | |__ src 60 | |__ admin 61 | |__ components 62 | |__ HelloWorld.vue 63 | |__ config 64 | |__ index.js 65 | |__ pages 66 | |__ index 67 | |__ App.vue 68 | |__ favicon.ico 69 | |__ index.html 70 | |__ index.js 71 | |__ router 72 | |__ index.js 73 | |__ global 74 | |__ assets 75 | |__ logo.png 76 | |__ master 77 | |__ components 78 | |__ HelloWorld.vue 79 | |__ config 80 | |__ index.js 81 | |__ pages 82 | |__ index 83 | |__ App.vue 84 | |__ favicon.ico 85 | |__ index.html 86 | |__ index.js 87 | |__ router 88 | |__ index.js 89 | |__ static 90 | |__ .gitkeep 91 | |__ test 92 | |__ .babelrc 93 | |__ .editorconfig 94 | |__ .gitignore 95 | |__ .postcssrc.js 96 | |__ package-lock.json 97 | |__ package.json 98 | |__ README.md 99 | ``` 100 | --- 101 | ### 使用 102 | #### git clone https://github.com/augustVino/vue-multi-project.git 103 | #### cd vue-multi-project 104 | 105 | ``` bash 106 | # clone 107 | git clone https://github.com/augustVino/vue-multi-project.git 108 | 109 | cd vue-multi-project 110 | 111 | # install dependencies 112 | npm install 113 | 114 | # serve with hot reload at localhost:8080 115 | npm run dev 116 | 117 | # build for production with minification 118 | npm run build 119 | 120 | # run unit tests 121 | npm run unit 122 | 123 | # run all tests 124 | npm test 125 | ``` 126 | 127 | ### 实现过程 128 | > #### 1.首先进入项目根路径中的 config 文件夹,新增两个文件( multipage.js、multipage.project.js ),分别是配置多页面和多项目切换的。 129 | > #### 2.然后修改 config 文件夹下的 index.js 文件。 130 | > #### 3.修改 build 文件夹下的 webpack.base.conf.js 、webpack.dev.conf.js 、webpack.prod.conf.js 。 131 | 132 | ### 工程简介 133 | > #### 1.有两个项目(admin和master),src路径下的另一个文件夹 global 是存放一些项目公共文件(如组件或工具函数),此处存放的是vue官方的logo图片,两个项目都用到了。 134 | > #### 2. 每个项目中都有个 pages 文件夹,用来存放项目入口文件。 135 | > #### 3. 每个项目都有独自的 config 文件,文件内容: 136 | ``` bash 137 | 'use strict' 138 | module.exports = { 139 | name : 'admin', //文件夹名称 和最后打包后文件夹的名称 /src/admin/ 140 | } 141 | ``` 142 | > #### 4. 切换项目:在根目录的config文件夹找到multipage.project.js文件打开,,切换到对应的项目进行开发打包。打包后也会对应生成独立的文件夹。 143 | -------------------------------------------------------------------------------- /build/build.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | // 检查npm和node的版本 3 | require('./check-versions')() 4 | // 设置环境变量NODE_ENV的值是production 5 | process.env.NODE_ENV = 'production' 6 | // 终端的spinner 7 | const ora = require('ora') 8 | // node.js版本的rm -rf 9 | const rm = require('rimraf') 10 | // 引入path模块 11 | const path = require('path') 12 | // 引入显示终端颜色模块 13 | const chalk = require('chalk') 14 | // 引入webpack模块 15 | const webpack = require('webpack') 16 | // 引入基本的配置文件 17 | const config = require('../config') 18 | // 引入webpack在production环境下的配置文件 19 | const webpackConfig = require('./webpack.prod.conf') 20 | 21 | const spinner = ora('building for production...') 22 | spinner.start() 23 | 24 | // 删除打包目标目录下的文件 25 | rm(path.join(config.build.assetsRoot, config.build.assetsSubDirectory), err => { 26 | if (err) throw err 27 | // 进行打包 28 | webpack(webpackConfig, (err, stats) => { 29 | // 打包完成 30 | spinner.stop() 31 | if (err) throw err 32 | // 输出打包的状态 33 | process.stdout.write(stats.toString({ 34 | colors: true, 35 | modules: false, 36 | children: false, // If you are using ts-loader, setting this to true will make TypeScript errors show up during build. 37 | chunks: false, 38 | chunkModules: false 39 | }) + '\n\n') 40 | 41 | // 如果打包出现错误 42 | if (stats.hasErrors()) { 43 | console.log(chalk.red(' Build failed with errors.\n')) 44 | process.exit(1) 45 | } 46 | 47 | // 打包完成 48 | console.log(chalk.cyan(' Build complete.\n')) 49 | console.log(chalk.yellow( 50 | ' Tip: built files are meant to be served over an HTTP server.\n' + 51 | ' Opening index.html over file:// won\'t work.\n' 52 | )) 53 | }) 54 | }) 55 | -------------------------------------------------------------------------------- /build/check-versions.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | // 在终端为不同字体显示不同的风格 3 | const chalk = require('chalk') 4 | // 解析npm包的version 5 | const semver = require('semver') 6 | // 引入package.json文件 7 | const packageConfig = require('../package.json') 8 | // node版本的uninx shell命令 9 | const shell = require('shelljs') 10 | 11 | // 执行命令的函数 12 | function exec (cmd) { 13 | return require('child_process').execSync(cmd).toString().trim() 14 | } 15 | 16 | const versionRequirements = [ 17 | { 18 | name: 'node', 19 | // node的版本 20 | // process.version就是node的版本 21 | // semver.clean('v8.8.0') => 8.8.0 22 | currentVersion: semver.clean(process.version), 23 | // package.json中定义的node版本的范围 24 | versionRequirement: packageConfig.engines.node 25 | } 26 | ] 27 | 28 | // 相当于 which npm 29 | if (shell.which('npm')) { 30 | // 如果npm命令存在的话 31 | versionRequirements.push({ 32 | name: 'npm', 33 | // 检查npm的版本 => 5.4.2 34 | currentVersion: exec('npm --version'), 35 | // package.json中定义的npm版本 36 | versionRequirement: packageConfig.engines.npm 37 | }) 38 | } 39 | 40 | module.exports = function () { 41 | const warnings = [] 42 | 43 | for (let i = 0; i < versionRequirements.length; i++) { 44 | const mod = versionRequirements[i] 45 | 46 | // semver.satisfies()进行版本之间的比较 47 | if (!semver.satisfies(mod.currentVersion, mod.versionRequirement)) { 48 | // 如果现有的npm或者node的版本比定义的版本低,则生成一段警告 49 | warnings.push(mod.name + ': ' + 50 | chalk.red(mod.currentVersion) + ' should be ' + 51 | chalk.green(mod.versionRequirement) 52 | ) 53 | } 54 | } 55 | 56 | if (warnings.length) { 57 | console.log('') 58 | console.log(chalk.yellow('To use this template, you must update following to modules:')) 59 | console.log() 60 | 61 | for (let i = 0; i < warnings.length; i++) { 62 | const warning = warnings[i] 63 | console.log(' ' + warning) 64 | } 65 | 66 | console.log() 67 | // 退出程序 68 | process.exit(1) 69 | } 70 | } 71 | -------------------------------------------------------------------------------- /build/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/augustVino/vue-multi-project/05f96068f4ab59aeceaff6ca187a0fdd3587dfad/build/logo.png -------------------------------------------------------------------------------- /build/utils.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | // 引入node.js的path模块,用于操作路径 3 | const path = require('path') 4 | // 引入模板的配置文件,下面就需要去这个文件中看看有什么基本的配置 5 | const config = require('../config') 6 | // 提取指定文件的插件,比如把css文件提取到一个文件中去 7 | const ExtractTextPlugin = require('extract-text-webpack-plugin') 8 | // 加载package.json文件 9 | const packageConfig = require('../package.json') 10 | 11 | // 生成编译输出的二级目录 12 | exports.assetsPath = function (_path) { 13 | const assetsSubDirectory = process.env.NODE_ENV === 'production' 14 | ? config.build.assetsSubDirectory 15 | : config.dev.assetsSubDirectory 16 | // path.posix是path模块跨平台的实现(不同平台的路径表示是不一样的) 17 | return path.posix.join(assetsSubDirectory, _path) 18 | } 19 | // 为不同的css预处理器提供一个统一的生成方式,也就是统一处理各种css类型的打包问题 20 | // 这个是为在vue文件中的style中使用的css类型 21 | exports.cssLoaders = function (options) { 22 | options = options || {} 23 | // 打包css模块 24 | const cssLoader = { 25 | loader: 'css-loader', 26 | options: { 27 | sourceMap: options.sourceMap 28 | } 29 | } 30 | // 编译postcss模块 31 | const postcssLoader = { 32 | // 使用postcss-loader来打包postcss模块 33 | loader: 'postcss-loader', 34 | // 配置source map 35 | options: { 36 | sourceMap: options.sourceMap 37 | } 38 | } 39 | 40 | // 创建loader加载器字符串,结合extract text插件使用 41 | /** 42 | * 43 | * @param {loader的名称} loader 44 | * @param {loader对应的options配置对象} loaderOptions 45 | */ 46 | function generateLoaders (loader, loaderOptions) { 47 | // 通过usePostCSS 来标明是否使用了postcss 48 | const loaders = options.usePostCSS ? [cssLoader, postcssLoader] : [cssLoader] 49 | // 如果指定了具体的loader的名称 50 | if (loader) { 51 | // 向loaders的数组中添加该loader对应的加载器 52 | // 一个很重要的地方就是,一个数组中的loader加载器,是从右向左执行的。 53 | loaders.push({ 54 | // loader加载器的名称 55 | loader: loader + '-loader', 56 | // 对应的加载器的配置对象 57 | options: Object.assign({}, loaderOptions, { 58 | sourceMap: options.sourceMap 59 | }) 60 | }) 61 | } 62 | 63 | // 如果明确指定了需要提取静态文件,则使用 64 | // ExtractTextPlugin.extract({})来包裹我们的各种css处理器。 65 | if (options.extract) { 66 | return ExtractTextPlugin.extract({ 67 | use: loaders, 68 | // fallback这个选项我们可以这样理解 69 | // webpack默认会按照loaders中的加载器从右向左调用编译各种css类型文件。如果一切顺利,在loaders中的 70 | // 各个加载器运行结束之后就会把css文件导入到规定的文件中去,如果不顺利,则继续使用vue-style-loader来处理 71 | // css文件 72 | fallback: 'vue-style-loader' 73 | }) 74 | } else { 75 | // 如果没有提取行为,则最后再使用vue-style-loader处理css 76 | return ['vue-style-loader'].concat(loaders) 77 | } 78 | } 79 | 80 | // https://vue-loader.vuejs.org/en/configurations/extract-css.html 81 | return { 82 | // css-loader 83 | css: generateLoaders(), 84 | // postcss-loader 85 | postcss: generateLoaders(), 86 | // less-loader 87 | less: generateLoaders('less'), 88 | // sass-loader 后面的选项表明sass使用的是缩进的愈发 89 | sass: generateLoaders('sass', { indentedSyntax: true }), 90 | // scss-loader 91 | scss: generateLoaders('sass'), 92 | // stylus-loader stylus文件有两种后缀名.stylus和styl 93 | stylus: generateLoaders('stylus'), 94 | // stylus-loader 95 | styl: generateLoaders('stylus') 96 | } 97 | } 98 | 99 | // 使用这个函数,为那些独立的style文件创建加载器配置。 100 | exports.styleLoaders = function (options) { 101 | // 保存到加载器配置的变量 102 | const output = [] 103 | // 获取所有css文件类型的loaders 104 | const loaders = exports.cssLoaders(options) 105 | 106 | for (const extension in loaders) { 107 | const loader = loaders[extension] 108 | // 生成对应的loader配置 109 | output.push({ 110 | test: new RegExp('\\.' + extension + '$'), 111 | use: loader 112 | }) 113 | } 114 | 115 | return output 116 | } 117 | 118 | exports.createNotifierCallback = () => { 119 | // node-notifier是一个跨平台的包,以类似浏览器的通知的形式展示信息。 120 | const notifier = require('node-notifier') 121 | 122 | return (severity, errors) => { 123 | // 只展示错误的信息 124 | if (severity !== 'error') return 125 | 126 | const error = errors[0] 127 | const filename = error.file && error.file.split('!').pop() 128 | 129 | // 需要展示的错误信息的内容 130 | notifier.notify({ 131 | // 通知的标题 132 | title: packageConfig.name, 133 | // 通知的主体内容 134 | message: severity + ': ' + error.name, 135 | // 副标题 136 | subtitle: filename || '', 137 | // 通知展示的icon 138 | icon: path.join(__dirname, 'logo.png') 139 | }) 140 | } 141 | } 142 | -------------------------------------------------------------------------------- /build/vue-loader.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const utils = require('./utils') 3 | const config = require('../config') 4 | // 设置是不是生产环境 5 | const isProduction = process.env.NODE_ENV === 'production' 6 | // 根据不同的环境,引入不同的source map配置文件 7 | const sourceMapEnabled = isProduction 8 | ? config.build.productionSourceMap 9 | : config.dev.cssSourceMap 10 | 11 | module.exports = { 12 | // vue文件中的css loader配置 13 | loaders: utils.cssLoaders({ 14 | sourceMap: sourceMapEnabled, 15 | // 生产环境下就会把css文件抽取到一个独立的文件中 16 | extract: isProduction 17 | }), 18 | // css source map文件的配置 19 | cssSourceMap: sourceMapEnabled, 20 | // css source map文件缓存控制变量 21 | cacheBusting: config.dev.cacheBusting, 22 | transformToRequire: { 23 | video: ['src', 'poster'], 24 | source: 'src', 25 | img: 'src', 26 | image: 'xlink:href' 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /build/webpack.base.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const path = require('path') 3 | const utils = require('./utils') 4 | const config = require('../config') 5 | const vueLoaderConfig = require('./vue-loader.conf') 6 | 7 | const multipage = require('../config/multipage'); //多页 8 | 9 | // 生成相对于根目录的绝对路径 10 | function resolve (dir) { 11 | return path.join(__dirname, '..', dir) 12 | } 13 | 14 | 15 | 16 | // 下面就是webpack基本的配置信息(可以立即生成是开发环境和生产环境的公共的配置) 17 | module.exports = { 18 | // webpack解析文件时候的根目录(如果把webpack.config.js放在了项目的根目录下面,这个配置可以省略) 19 | context: path.resolve(__dirname, '../'), 20 | // 指定项目的入口文件(用multipage.getEntries()方法替换原有设置) 21 | // entry: { 22 | // app: './src/main.js' 23 | // }, 24 | entry : multipage.getEntries(), 25 | // 项目的输出配置 26 | output: { 27 | // 项目build的时候,生成的文件的存放路径(这里的路径是../dist) 28 | path: config.build.assetsRoot, 29 | // 生成文件的名称 30 | filename: '[name].js', 31 | // 输出解析文件的目录,url相对于HTML页面(生成的html文件中,css和js等静态文件的url前缀) 32 | publicPath: process.env.NODE_ENV === 'production' 33 | ? config.build.assetsPublicPath 34 | : config.dev.assetsPublicPath 35 | }, 36 | // 配置模块解析时候的一些选项 37 | resolve: { 38 | // 指定哪些类型的文件可以引用的时候省略后缀名 39 | extensions: ['.js', '.vue', '.json'], 40 | // 别名,在引入文件的时候可以使用 41 | alias: { 42 | 'vue$': 'vue/dist/vue.esm.js', 43 | // 可以在引入文件的时候使用@符号引入src文件夹中的文件 44 | '@': resolve('src/'+multipage.project.name), //当前项目 src/项目名称/ 45 | '@g': resolve('src/global'), //当前项目 src/global/ 全局组件 46 | '@cs': resolve('src/'+multipage.project.name+'/components'), //组件路径简写 47 | } 48 | }, 49 | 50 | // 下面是针对具体的模块进行的具体的配置 51 | // 下面的配置语法采用的是 version >= @2 的版本 52 | module: { 53 | // rules是一个数组,其中每一个元素都是一个对象,这个对象是针对具体类型的文件进行的配置 54 | rules: [ 55 | // .vue文件的配置 56 | { 57 | // 这个属性是一个正则表达式,用于匹配文件。这里匹配的是.vue文件 58 | test: /\.vue$/, 59 | // 指定该种类型文件的加载器名称 60 | loader: 'vue-loader', 61 | // 针对此加载器的具体配置 62 | // 针对前面的分析,这个配置对象中包含了各种css类型文件的配置,css source map的配置 以及一些transform的配置 63 | options: vueLoaderConfig 64 | }, 65 | // .js文件的配置 66 | { 67 | test: /\.js$/, 68 | // js文件的处理主要使用的是babel-loader。在这里没有指定具体的编译规则,babel-loader会自动 69 | // 读取根目录下面的.babelrc中的babel配置用于编译js文件 70 | /** 71 | * { 72 | * // 使用的预设 73 | "presets": [ 74 | // babel-preset-env: 根据你所支持的环境自动决定具体类型的babel插件 75 | ["env", { 76 | // modules设置为false,不会转换module 77 | "modules": false 78 | }], 79 | // babel-preset-stage-2: 可以使用所有>=stage2语法 80 | "stage-2" 81 | ], 82 | // 使用的插件 83 | // babel-plugin-transform-runtime: 只会对es6的语法进行转换而不会对新的api进行转换 84 | // 如果需要支持新的api,请引入babel-polyfill 85 | "plugins": ["transform-runtime"] 86 | } 87 | */ 88 | loader: 'babel-loader', 89 | // 指定需要进行编译的文件的路径 90 | // 这里表示只对src和test文件夹中的文件进行编译 91 | include: [resolve('src'), resolve('test') ,resolve('node_modules/webpack-dev-server/client')] 92 | }, 93 | { 94 | // 对图片资源进行编译的配置 95 | // 指定文件的类型 96 | test: /\.(png|jpe?g|gif|svg)(\?.*)?$/, 97 | // 使用url-loader进行文件资源的编译 98 | loader: 'url-loader', 99 | // url-loader的配置选项 100 | options: { 101 | // 文件的大小小于10000字节(10kb)的时候会返回一个dataUrl 102 | limit: 10000, 103 | // 生成的文件的保存路径和后缀名称 104 | name: utils.assetsPath('img/[name].[hash:7].[ext]') 105 | } 106 | }, 107 | // 对视频文件进行打包编译 108 | { 109 | test: /\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/, 110 | loader: 'url-loader', 111 | options: { 112 | limit: 10000, 113 | name: utils.assetsPath('media/[name].[hash:7].[ext]') 114 | } 115 | }, 116 | // 对字体文件进行打包编译 117 | { 118 | test: /\.(woff2?|eot|ttf|otf)(\?.*)?$/, 119 | loader: 'url-loader', 120 | options: { 121 | limit: 10000, 122 | name: utils.assetsPath('fonts/[name].[hash:7].[ext]') 123 | } 124 | } 125 | ] 126 | }, 127 | // 这些选项用于配置polyfill或mock某些node.js全局变量和模块。 128 | // 这可以使最初为nodejs编写的代码可以在浏览器端运行 129 | node: { 130 | // 这个配置是一个对象,其中的每个属性都是nodejs全局变量或模块的名称 131 | // prevent webpack from injecting useless setImmediate polyfill because Vue 132 | // source contains it (although only uses it if it's native). 133 | // false表示什么都不提供。如果获取此对象的代码,可能会因为获取不到此对象而触发ReferenceError错误 134 | setImmediate: false, 135 | // prevent webpack from injecting mocks to Node native modules 136 | // that does not make sense for the client 137 | // 设置成empty则表示提供一个空对象 138 | dgram: 'empty', 139 | fs: 'empty', 140 | net: 'empty', 141 | tls: 'empty', 142 | child_process: 'empty' 143 | } 144 | } 145 | -------------------------------------------------------------------------------- /build/webpack.dev.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const path = require('path') 3 | // 首先引入的是一些工具方法,下面我们就需要去util文件种看一下有哪些对应的工具方法 4 | const utils = require('./utils') 5 | // 引入webpack模块 6 | const webpack = require('webpack') 7 | // 引入配置文件 8 | // 这个配置文件中包含了一些dev和production环境的基本配置 9 | const config = require('../config') 10 | // 引入webpack-merge模块。这个模块用于把多个webpack配置合并成一个配置,后面的配置会覆盖前面的配置。 11 | const merge = require('webpack-merge') 12 | // 引入webpack的基本设置,这个设置文件包含了开发环境和生产环境的一些公共配置 13 | const baseWebpackConfig = require('./webpack.base.conf') 14 | // 15 | const CopyWebpackPlugin = require('copy-webpack-plugin') 16 | 17 | // 用于生成html文件的插件 18 | const HtmlWebpackPlugin = require('html-webpack-plugin') 19 | // 这个插件能够更好的在终端看到webpack运行时的错误和警告等信息。可以提升开发体验。 20 | const FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin') 21 | // 查找一个未使用的端口 22 | const portfinder = require('portfinder') 23 | 24 | //新增多页配置 25 | const multipage = require('../config/multipage'); 26 | 27 | 28 | function resolve (dir) { 29 | return path.join(__dirname, '..', dir) 30 | } 31 | 32 | // 获取host环境变量,用于配置开发环境域名 33 | const HOST = process.env.HOST 34 | // 获取post环境变量,用于配置开发环境时候的端口号 35 | const PORT = process.env.PORT && Number(process.env.PORT) 36 | 37 | // 开发环境的完整的配置文件, 38 | const devWebpackConfig = merge(baseWebpackConfig, { 39 | module: { 40 | // 为那些独立的css类型文件添加loader配置(没有写在vue文件的style标签中的样式) 41 | rules: utils.styleLoaders({ sourceMap: config.dev.cssSourceMap, usePostCSS: true }) 42 | }, 43 | // 开发环境使用'eval-source-map'模式的source map 44 | // 因为速度快 45 | devtool: config.dev.devtool, 46 | 47 | // these devServer options should be customized in /config/index.js 48 | // 下面是对webpack-dev-server选项的基本配置,这些配置信息,我们可以在/config/index.js 49 | // 文件中进行自定义配置。 50 | devServer: { 51 | // 用于配置在开发工具的控制台中显示的日志级别 52 | // 注意这个不是对bundle的错误和警告的配置,而是对它生成之前的消息的配置 53 | clientLogLevel: 'warning', 54 | // 表示当使用html5的history api的时候,任意的404响应都需要被替代为index.html 55 | // historyApiFallback: true, 56 | historyApiFallback: { 57 | // rewrites: [ 58 | // { from: /.*/, to: path.posix.join(config.dev.assetsPublicPath, 'index.html') }, 59 | // ], 60 | // 此处替换为multipage.getHistoryRewrites()方法 61 | rewrites : multipage.getHistoryRewrites() 62 | }, 63 | // 启用webpack的热替换特性 64 | hot: true, 65 | // 一切服务都需要使用gzip压缩 66 | // 可以在js,css等文件的response header中发现有Content-Encoding:gzip响应头 67 | compress: true, 68 | contentBase: false, // since we use CopyWebpackPlugin. 69 | // 指定使用一个 host。默认是 localhost 70 | // 如果希望服务器外部可以访问(通过我们电脑的ip地址和端口号访问我们的应用) 71 | // 可以指定0.0.0.0 72 | host: HOST || config.dev.host, 73 | // 指定要监听请求的端口号 74 | port: PORT || config.dev.port, 75 | // 是否自动打开浏览器 76 | open: config.dev.autoOpenBrowser, 77 | // 当编译出现错误的时候,是否希望在浏览器中展示一个全屏的蒙层来展示错误信息 78 | overlay: config.dev.errorOverlay 79 | // 表示只显示错误信息而不显示警告信息 80 | // 如果两者都希望显示,则把这两项都设置为true 81 | ? { warnings: false, errors: true } 82 | // 设置为false则表示啥都不显示 83 | : false, 84 | // 指定webpack-dev-server的根目录,这个目录下的所有的文件都是能直接通过浏览器访问的 85 | // 推荐和output.publicPath设置为一致 86 | publicPath: config.dev.assetsPublicPath, 87 | // 配置代理,这样我们就可以跨域访问某些接口 88 | // 我们访问的接口,如果符合这个选项的配置,就会通过代理服务器转发我们的请求 89 | proxy: config.dev.proxyTable, 90 | // 启用 quiet 后,除了初始启动信息之外的任何内容都不会被打印到控制台。这也意味着来自 webpack 的错误或警告在控制台不可见。 91 | quiet: true, // necessary for FriendlyErrorsPlugin 92 | // 与监视文件相关的控制选项。 93 | watchOptions: { 94 | // 如果这个选项为true,会以轮询的方式检查我们的文件的变动,效率不好 95 | poll: config.dev.poll, 96 | } 97 | }, 98 | plugins: [ 99 | // 创建一个在编译时可以配置的全局变量 100 | new webpack.DefinePlugin({ 101 | 'process.env': require('../config/dev.env') 102 | }), 103 | // 启用热替换模块 104 | // 记住,我们永远不要再生产环境中使用hmr 105 | new webpack.HotModuleReplacementPlugin(), 106 | // 这个插件的主要作用就是在热加载的时候直接返回更新文件的名称,而不是文件的id 107 | new webpack.NamedModulesPlugin(), // HMR shows correct file names in console on update. 108 | // 使用这个插件可以在编译出错的时候来跳过输出阶段,这样可以确保输出资源不会包含错误。 109 | new webpack.NoEmitOnErrorsPlugin(), 110 | // 这个插件主要是生成一个html文件 111 | // new HtmlWebpackPlugin({ 112 | // // 生成的html文件的名称 113 | // filename: 'index.html', 114 | // // 使用的模板的名称 115 | // template: 'index.html', 116 | // // 将所有的静态文件都插入到body文件的末尾 117 | // inject: true 118 | // }), 119 | // copy custom static assets 120 | new CopyWebpackPlugin([ 121 | { 122 | from: path.resolve(__dirname, '../static'), 123 | to: config.dev.assetsSubDirectory, 124 | ignore: ['.*'] 125 | } 126 | ]) 127 | ] 128 | }) 129 | 130 | module.exports = new Promise((resolve, reject) => { 131 | portfinder.basePort = process.env.PORT || config.dev.port 132 | portfinder.getPort((err, port) => { 133 | if (err) { 134 | reject(err) 135 | } else { 136 | // publish the new Port, necessary for e2e tests 137 | process.env.PORT = port 138 | // add port to devServer config 139 | devWebpackConfig.devServer.port = port 140 | 141 | // Add FriendlyErrorsPlugin 142 | devWebpackConfig.plugins.push(new FriendlyErrorsPlugin({ 143 | compilationSuccessInfo: { 144 | messages: [`Your application is running here: http://${devWebpackConfig.devServer.host}:${port}`], 145 | }, 146 | onErrors: config.dev.notifyOnErrors 147 | ? utils.createNotifierCallback() 148 | : undefined 149 | })) 150 | 151 | resolve(devWebpackConfig) 152 | } 153 | }) 154 | }) 155 | 156 | 157 | // HtmlWebpackPlugin 替换HtmlWebpackPlugin方法 158 | var devHtmlList = multipage.getDevHtmlList(); 159 | devHtmlList.forEach(function(item){ 160 | devWebpackConfig.plugins.push(item) 161 | }) -------------------------------------------------------------------------------- /build/webpack.prod.conf.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | // 引入path模块 4 | const path = require('path') 5 | // 引入工具方法 6 | const utils = require('./utils') 7 | // 引入webpack模块 8 | const webpack = require('webpack') 9 | // 引入基本的配置 10 | const config = require('../config') 11 | // 引入webpack-merge模块 12 | const merge = require('webpack-merge') 13 | // 引入开发环境和生产环境公共的配置 14 | const baseWebpackConfig = require('./webpack.base.conf') 15 | // 引入copy-webpack-plugin模块 16 | // 这个模块主要用于在webpack中拷贝文件和文件夹 17 | const CopyWebpackPlugin = require('copy-webpack-plugin') 18 | // 引入html-webpack-plugin插件 19 | // 这个插件主要是用于基于模版生成html文件的 20 | const HtmlWebpackPlugin = require('html-webpack-plugin') 21 | // 引入extract-text-webpack-plugin插件 22 | // 这个插件主要是用于将入口中所有的chunk,移到独立的分离的css文件中 23 | const ExtractTextPlugin = require('extract-text-webpack-plugin') 24 | // 引入optimize-css-assets-webpack-plugin插件 25 | // 这个插件主要是用于压缩css模块的 26 | const OptimizeCSSPlugin = require('optimize-css-assets-webpack-plugin') 27 | // 引入uglifyjs-webpack-plugin插件 28 | // 这个插件主要是用于压缩js文件的 29 | const UglifyJsPlugin = require('uglifyjs-webpack-plugin') 30 | // 新增多页配置 31 | const multipage = require('../config/multipage'); //多页 32 | 33 | function resolve (dir) { 34 | return path.join(__dirname, '..', dir) 35 | } 36 | // 引入用于生产环境的一些基本变量 37 | const env = process.env.NODE_ENV === 'testing' 38 | ? require('../config/test.env') 39 | : require('../config/prod.env') 40 | 41 | // 合并公共配置和生产环境独有的配置并返回一个用于生产环境的webpack配置文件 42 | const webpackConfig = merge(baseWebpackConfig, { 43 | // 用于生产环境的一些loader配置 44 | module: { 45 | rules: utils.styleLoaders({ 46 | sourceMap: config.build.productionSourceMap, 47 | // 在生产环境中使用extract选项,这样就会把thunk中的css代码抽离到一份独立的css文件中去 48 | extract: true, 49 | usePostCSS: true 50 | }) 51 | }, 52 | // 配置生产环境中使用的source map的形式。在这里,生产环境使用的是#source map的形式 53 | devtool: config.build.productionSourceMap ? config.build.devtool : false, 54 | output: { 55 | // build所产生的文件的存放的文件夹地址 56 | path: config.build.assetsRoot, 57 | // build之后的文件的名称 58 | // 这里[name]和[chunkhash]都是占位符 59 | // 其中[name]指的就是模块的名称 60 | // [chunkhash]chunk内容的hash字符串,长度为20 61 | filename: utils.assetsPath('js/[name].[chunkhash].js'), 62 | // [id]也是一个占位符,表示的是模块标识符(module identifier) 63 | chunkFilename: utils.assetsPath('js/[id].[chunkhash].js') 64 | }, 65 | plugins: [ 66 | // http://vuejs.github.io/vue-loader/en/workflow/production.html 67 | new webpack.DefinePlugin({ 68 | 'process.env': env 69 | }), 70 | // 压缩javascript的插件 71 | new UglifyJsPlugin({ 72 | // 压缩js的时候的一些基本配置 73 | uglifyOptions: { 74 | // 配置压缩的行为 75 | compress: { 76 | // 在删除未使用的变量等时,显示警告信息,默认就是false 77 | warnings: false 78 | } 79 | }, 80 | // 使用 source map 将错误信息的位置映射到模块(这会减慢编译的速度) 81 | // 而且这里不能使用cheap-source-map 82 | sourceMap: config.build.productionSourceMap, 83 | // 使用多进程并行运行和文件缓存来提高构建速度 84 | parallel: true 85 | }), 86 | // 提取css文件到一个独立的文件中去 87 | new ExtractTextPlugin({ 88 | // 提取之后css文件存放的地方 89 | // 其中[name]和[contenthash]都是占位符 90 | // [name]就是指模块的名称 91 | // [contenthash]根据提取文件的内容生成的 hash 92 | filename: utils.assetsPath('css/[name].[contenthash].css'), 93 | 94 | // 从所有额外的 chunk(additional chunk) 提取css内容 95 | // (默认情况下,它仅从初始chunk(initial chunk) 中提取) 96 | // 当使用 CommonsChunkPlugin 并且在公共 chunk 中有提取的 chunk(来自ExtractTextPlugin.extract)时 97 | // 这个选项需要设置为true 98 | allChunks: true, 99 | }), 100 | // 使用这个插件压缩css,主要是因为,对于不同组件中相同的css可以剔除一部分 101 | new OptimizeCSSPlugin({ 102 | // 这个选项的所有配置都会传递给cssProcessor 103 | // cssProcessor使用这些选项决定压缩的行为 104 | cssProcessorOptions: config.build.productionSourceMap 105 | ? { safe: true, map: { inline: false } } 106 | : { safe: true } 107 | }), 108 | // 创建一个html文件 109 | // new HtmlWebpackPlugin({ 110 | // // 生成的文件的名称 111 | // filename: config.build.index, 112 | // // 使用的模板的名称 113 | // template: 'index.html', 114 | // // 把script和link标签放在body底部 115 | // inject: true, 116 | // favicon: resolve('favicon.ico'), 117 | // title: '车联网管理平台', 118 | // // 配置html的压缩行为 119 | // minify: { 120 | // // 移除注释 121 | // removeComments: true, 122 | // // 去除空格和换行 123 | // collapseWhitespace: true, 124 | // // 尽可能移除属性中的引号和空属性 125 | // removeAttributeQuotes: true 126 | // // more options: 127 | // // https://github.com/kangax/html-minifier#options-quick-reference 128 | // }, 129 | // // 控制chunks的顺序,这里表示按照依赖关系进行排序 130 | // // 也可以是一个函数,自己定义排序规则 131 | // chunksSortMode: 'dependency' 132 | // }), 133 | // 根据模块的相对路径生成一个四位数的hash作为模块id 134 | new webpack.HashedModuleIdsPlugin(), 135 | // webpack2处理过的每一个模块都会使用一个函数进行包裹 136 | // 这样会带来一个问题:降低浏览器中JS执行效率,这主要是闭包函数降低了JS引擎解析速度。 137 | // webpack3中,通过下面这个插件就能够将一些有联系的模块, 138 | // 放到一个闭包函数里面去,通过减少闭包函数数量从而加快JS的执行速度。 139 | new webpack.optimize.ModuleConcatenationPlugin(), 140 | // 这个插件用于提取多入口chunk的公共模块 141 | // 通过将公共模块提取出来之后,最终合成的文件能够在最开始的时候加载一次 142 | // 然后缓存起来供后续使用,这会带来速度上的提升。 143 | new webpack.optimize.CommonsChunkPlugin({ 144 | // 这是 common chunk 的名称 145 | name: 'vendor', 146 | // 把所有从mnode_modules中引入的文件提取到vendor中 147 | minChunks (module) { 148 | // any required modules inside node_modules are extracted to vendor 149 | return ( 150 | module.resource && 151 | /\.js$/.test(module.resource) && 152 | module.resource.indexOf( 153 | path.join(__dirname, '../node_modules') 154 | ) === 0 155 | ) 156 | } 157 | }), 158 | // 为了将项目中的第三方依赖代码抽离出来,官方文档上推荐使用这个插件,当我们在项目里实际使用之后, 159 | // 发现一旦更改了 app.js 内的代码,vendor.js 的 hash 也会改变,那么下次上线时, 160 | // 用户仍然需要重新下载 vendor.js 与 app.js——这样就失去了缓存的意义了。所以第二次new就是解决这个问题的 161 | // 参考:https://github.com/DDFE/DDFE-blog/issues/10 162 | new webpack.optimize.CommonsChunkPlugin({ 163 | name: 'manifest', 164 | minChunks: Infinity 165 | }), 166 | // This instance extracts shared chunks from code splitted chunks and bundles them 167 | // in a separate chunk, similar to the vendor chunk 168 | // see: https://webpack.js.org/plugins/commons-chunk-plugin/#extra-async-commons-chunk 169 | new webpack.optimize.CommonsChunkPlugin({ 170 | name: 'app', 171 | async: 'vendor-async', 172 | children: true, 173 | minChunks: 3 174 | }), 175 | 176 | // 拷贝静态资源到build文件夹中 177 | new CopyWebpackPlugin([ 178 | { 179 | // 定义要拷贝的资源的源目录 180 | from: path.resolve(__dirname, '../static'), 181 | // 定义要拷贝的资源的目标目录 182 | to: config.build.assetsSubDirectory, 183 | // 忽略拷贝指定的文件,可以使用模糊匹配 184 | ignore: ['.*'] 185 | } 186 | ]) 187 | ] 188 | }) 189 | 190 | if (config.build.productionGzip) { 191 | // 如果开启了生产环境的gzip 192 | const CompressionWebpackPlugin = require('compression-webpack-plugin') 193 | 194 | webpackConfig.plugins.push( 195 | new CompressionWebpackPlugin({ 196 | // 目标资源的名称 197 | // [path]会被替换成原资源路径 198 | // [query]会被替换成原查询字符串 199 | asset: '[path].gz[query]', 200 | // gzip算法 201 | // 这个选项可以配置成zlib模块中的各个算法 202 | // 也可以是(buffer, cb) => cb(buffer) 203 | algorithm: 'gzip', 204 | // 处理所有匹配此正则表达式的资源 205 | test: new RegExp( 206 | '\\.(' + 207 | config.build.productionGzipExtensions.join('|') + 208 | ')$' 209 | ), 210 | // 只处理比这个值大的资源 211 | threshold: 10240, 212 | // 只有压缩率比这个值小的资源才会被处理 213 | minRatio: 0.8 214 | }) 215 | ) 216 | } 217 | 218 | if (config.build.bundleAnalyzerReport) { 219 | // 如果需要生成一分bundle报告,则需要使用下面的这个插件 220 | const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin 221 | webpackConfig.plugins.push(new BundleAnalyzerPlugin()) 222 | } 223 | 224 | module.exports = webpackConfig 225 | 226 | 227 | // HtmlWebpackPlugin 替换HtmlWebpackPlugin方法 228 | var prodHtmlList = multipage.getProdHtmlList(); 229 | prodHtmlList.forEach(function(item){ 230 | webpackConfig.plugins.push(item) 231 | }) -------------------------------------------------------------------------------- /config/dev.env.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const merge = require('webpack-merge') 3 | const prodEnv = require('./prod.env') 4 | 5 | module.exports = merge(prodEnv, { 6 | NODE_ENV: '"development"' 7 | }) 8 | -------------------------------------------------------------------------------- /config/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | 3 | // 这个文件主要是对开发环境和生产环境的一个基本的配置 4 | 5 | const path = require('path') 6 | 7 | /** 8 | * 注意 1 (新增) 9 | */ 10 | const projectInfo = require('./multipage.project') //项目信息 11 | 12 | module.exports = { 13 | // 开发环境的一个基本配置 14 | dev: { 15 | 16 | // Paths 17 | assetsSubDirectory: 'static', 18 | assetsPublicPath: '/', 19 | proxyTable: {}, 20 | 21 | // Various Dev Server settings 22 | host: 'localhost', // can be overwritten by process.env.HOST 23 | port: 8080, // can be overwritten by process.env.PORT, if port is in use, a free one will be determined 24 | autoOpenBrowser: false, 25 | errorOverlay: true, 26 | notifyOnErrors: true, 27 | poll: false, // https://webpack.js.org/configuration/dev-server/#devserver-watchoptions- 28 | 29 | 30 | /** 31 | * Source Maps 32 | */ 33 | 34 | // https://webpack.js.org/configuration/devtool/#development 35 | devtool: 'cheap-module-eval-source-map', 36 | 37 | // If you have problems debugging vue-files in devtools, 38 | // set this to false - it *may* help 39 | // https://vue-loader.vuejs.org/en/options.html#cachebusting 40 | cacheBusting: true, 41 | 42 | cssSourceMap: true 43 | }, 44 | 45 | build: { 46 | // Template for index.html 47 | // index: path.resolve(__dirname, '../dist/index.html'), 48 | // html文件的生成的地方 49 | index: path.resolve(__dirname, '../dist/'+projectInfo.name+'/index.html'), 50 | 51 | // 编译生成的文件的目录 52 | assetsRoot: path.resolve(__dirname, '../dist/'+projectInfo.name), 53 | // Paths 54 | // assetsRoot: path.resolve(__dirname, '../dist'), 55 | assetsSubDirectory: 'static', 56 | assetsPublicPath: '/', 57 | 58 | /** 59 | * Source Maps 60 | */ 61 | 62 | productionSourceMap: true, 63 | // https://webpack.js.org/configuration/devtool/#production 64 | devtool: '#source-map', 65 | 66 | // Gzip off by default as many popular static hosts such as 67 | // Surge or Netlify already gzip all static assets for you. 68 | // Before setting to `true`, make sure to: 69 | // npm install --save-dev compression-webpack-plugin 70 | productionGzip: false, 71 | productionGzipExtensions: ['js', 'css'], 72 | 73 | // Run the build command with an extra argument to 74 | // View the bundle analyzer report after build finishes: 75 | // `npm run build --report` 76 | // Set to `true` or `false` to always turn it on or off 77 | bundleAnalyzerReport: process.env.npm_config_report 78 | } 79 | } 80 | -------------------------------------------------------------------------------- /config/multipage.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const path = require('path') 3 | const fs = require("fs") 4 | const config = require('./index') 5 | const HtmlWebpackPlugin = require('html-webpack-plugin') 6 | 7 | //项目信息 8 | const projectInfo = require('./multipage.project') 9 | 10 | 11 | let baseURL = './src/'+projectInfo.name+'/pages'; //根路径 12 | let moduleList = getModuleList(baseURL); //页面 13 | 14 | function resolve (dir) { 15 | return path.join(__dirname, '..', dir) 16 | } 17 | //获取当前目录中文件 18 | function getModuleList(path){ 19 | let list = []; 20 | let fsList = fs.readdirSync(path); //所有目录 21 | fsList.forEach(function(item,index){ 22 | let module = { 23 | id : item, 24 | js : path+'/'+item+'/index.js', 25 | html : path+'/'+item+'/index.html', 26 | favicon : path+'/'+item+'/favicon.ico' 27 | } 28 | list.push(module); 29 | }) 30 | return list; 31 | } 32 | 33 | //webpack.base.conf.js entry 34 | function getEntries(){ 35 | //js入口数组 36 | let entries = {} 37 | //变量模块列表 38 | moduleList.forEach(function (item) { 39 | if(item.id){ 40 | entries[item.id] = item.js; 41 | } 42 | }) 43 | return entries; 44 | } 45 | 46 | //webpack.dev.conf.js HtmlWebpackPlugin 47 | function getDevHtmlList(){ 48 | //缓存dev的Html模板 49 | let devList = []; 50 | 51 | moduleList.forEach(function (item) { 52 | if(item.id){ 53 | let opts = { 54 | filename: item.html, 55 | template: item.html, 56 | favicon: resolve(item.favicon), 57 | inject: true, 58 | chunks: [item.id, "vendor", "manifest"] 59 | } 60 | devList.push(new HtmlWebpackPlugin(opts)); 61 | } 62 | }) 63 | return devList; 64 | } 65 | 66 | //webpack.dev.conf.js devServer.historyApiFallback.rewrites 67 | function getHistoryRewrites(){ 68 | //缓存dev的 historyApiFallback.rewrites 69 | let historyList = []; 70 | 71 | moduleList.forEach(function (item) { 72 | if(item.id){ 73 | if(item.id == 'index'){ 74 | let opts = { 75 | from: /.*/, 76 | to: path.posix.join(config.dev.assetsPublicPath, item.html) 77 | } 78 | historyList.push(opts); 79 | }else{ 80 | let opts = { 81 | from: new RegExp('^\/'+item.id+''), 82 | to: path.posix.join(config.dev.assetsPublicPath, item.html) 83 | } 84 | historyList.unshift(opts); 85 | } 86 | } 87 | }) 88 | 89 | return historyList; 90 | } 91 | 92 | //webpack.prod.conf.js HtmlWebpackPlugin 93 | function getProdHtmlList(){ 94 | //缓存dev的Html模板 95 | let prodList = []; 96 | 97 | moduleList.forEach(function (item) { 98 | if(item.id){ 99 | let filename; 100 | if(item.id == 'index'){ 101 | filename = config.build.index; 102 | }else{ 103 | filename = config.build.assetsRoot+'/'+item.id+'/index.html' 104 | } 105 | let opts = { 106 | filename: filename, 107 | template: item.html, 108 | favicon: resolve(item.favicon), 109 | inject: true, 110 | minify: { 111 | removeComments: true, 112 | collapseWhitespace: true, 113 | removeAttributeQuotes: true 114 | }, 115 | chunksSortMode: 'dependency' 116 | } 117 | prodList.push(new HtmlWebpackPlugin(opts)); 118 | } 119 | }) 120 | return prodList; 121 | } 122 | 123 | module.exports = { 124 | getEntries, 125 | getDevHtmlList, 126 | getHistoryRewrites, 127 | getProdHtmlList, 128 | project : projectInfo 129 | } 130 | -------------------------------------------------------------------------------- /config/multipage.project.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | // const project = require('../src/admin/config') //admin 3 | const project = require('../src/master/config') //master 4 | //const project = require('../src/traffic/config') //traffic 5 | 6 | module.exports = project; -------------------------------------------------------------------------------- /config/prod.env.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | module.exports = { 3 | NODE_ENV: '"production"' 4 | } 5 | -------------------------------------------------------------------------------- /config/test.env.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | const merge = require('webpack-merge') 3 | const devEnv = require('./dev.env') 4 | 5 | module.exports = merge(devEnv, { 6 | NODE_ENV: '"testing"' 7 | }) 8 | -------------------------------------------------------------------------------- /dist/admin/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/augustVino/vue-multi-project/05f96068f4ab59aeceaff6ca187a0fdd3587dfad/dist/admin/favicon.ico -------------------------------------------------------------------------------- /dist/admin/index.html: -------------------------------------------------------------------------------- 1 | vue-multi-project
-------------------------------------------------------------------------------- /dist/admin/static/css/index.3a947c1c1f207832148c696e78c1f1bd.css: -------------------------------------------------------------------------------- 1 | #app{font-family:Avenir,Helvetica,Arial,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;text-align:center;color:#2c3e50;margin-top:60px}h1[data-v-29427ef6],h2[data-v-29427ef6]{font-weight:400}ul[data-v-29427ef6]{list-style-type:none;padding:0}li[data-v-29427ef6]{display:inline-block;margin:0 10px}a[data-v-29427ef6]{color:#42b983} 2 | /*# sourceMappingURL=index.3a947c1c1f207832148c696e78c1f1bd.css.map */ -------------------------------------------------------------------------------- /dist/admin/static/css/index.3a947c1c1f207832148c696e78c1f1bd.css.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["index.3a947c1c1f207832148c696e78c1f1bd.css"],"names":[],"mappings":"AACA,KACE,8CAAoD,AACpD,mCAAoC,AACpC,kCAAmC,AACnC,kBAAmB,AACnB,cAAe,AACf,eAAiB,CAClB,AAED,wCACE,eAAoB,CACrB,AACD,oBACE,qBAAsB,AACtB,SAAW,CACZ,AACD,oBACE,qBAAsB,AACtB,aAAe,CAChB,AACD,mBACE,aAAe,CAChB","file":"index.3a947c1c1f207832148c696e78c1f1bd.css","sourcesContent":["\n#app {\n font-family: 'Avenir', Helvetica, Arial, sans-serif;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n text-align: center;\n color: #2c3e50;\n margin-top: 60px;\n}\n\nh1[data-v-29427ef6], h2[data-v-29427ef6] {\n font-weight: normal;\n}\nul[data-v-29427ef6] {\n list-style-type: none;\n padding: 0;\n}\nli[data-v-29427ef6] {\n display: inline-block;\n margin: 0 10px;\n}\na[data-v-29427ef6] {\n color: #42b983;\n}\n"]} -------------------------------------------------------------------------------- /dist/admin/static/js/index.3675151582d13b8e0c7e.js: -------------------------------------------------------------------------------- 1 | webpackJsonp([1],{"2kgF":function(t,i){},"4BF6":function(t,i){},akkm:function(t,i){t.exports=""},hfAa:function(t,i,c){"use strict";Object.defineProperty(i,"__esModule",{value:!0});var I=c("7+uW"),n={name:"App",data:function(){return{imgurl:c("akkm")}}},l={render:function(){var t=this.$createElement,i=this._self._c||t;return i("div",{attrs:{id:"app"}},[i("img",{attrs:{src:this.imgurl}}),this._v(" "),i("router-view")],1)},staticRenderFns:[]};var e=c("VU/8")(n,l,!1,function(t){c("4BF6")},null,null).exports,s=c("/ocq"),a={render:function(){var t=this.$createElement,i=this._self._c||t;return i("div",{staticClass:"hello"},[i("h1",[this._v(this._s(this.msg))])])},staticRenderFns:[]};var h=c("VU/8")({name:"HelloWorld",data:function(){return{msg:"欢迎来到admin项目"}}},a,!1,function(t){c("2kgF")},"data-v-29427ef6",null).exports;I.a.use(s.a);var k=new s.a({routes:[{path:"/",name:"HelloWorld",component:h}]});I.a.config.productionTip=!1,new I.a({el:"#app",router:k,components:{App:e},template:""})}},["hfAa"]); 2 | //# sourceMappingURL=index.3675151582d13b8e0c7e.js.map -------------------------------------------------------------------------------- /dist/admin/static/js/index.3675151582d13b8e0c7e.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["webpack:///./src/global/assets/logo.png","webpack:///src/admin/pages/index/App.vue","webpack:///./src/admin/pages/index/App.vue?a83c","webpack:///./src/admin/pages/index/App.vue","webpack:///./src/admin/components/HelloWorld.vue?d4f0","webpack:///./src/admin/components/HelloWorld.vue","webpack:///src/admin/components/HelloWorld.vue","webpack:///./src/admin/router/index.js","webpack:///./src/admin/pages/index/index.js"],"names":["module","exports","App","index_App","render","_h","this","$createElement","_c","_self","attrs","id","src","imgurl","_v","staticRenderFns","pages_index_App","__webpack_require__","normalizeComponent","ssrContext","components_HelloWorld","staticClass","_s","msg","admin_components_HelloWorld","HelloWorld_normalizeComponent","vue_esm","use","vue_router_esm","router","routes","path","name","component","config","productionTip","el","components","template"],"mappings":"mFAAAA,EAAAC,QAAA,yiSCQAC,QAEA,sCAIA,WCXAC,GADiBC,OAFjB,WAA0B,IAAaC,EAAbC,KAAaC,eAA0BC,EAAvCF,KAAuCG,MAAAD,IAAAH,EAAwB,OAAAG,EAAA,OAAiBE,OAAOC,GAAA,SAAYH,EAAA,OAAYE,OAAOE,IAAtHN,KAAsHO,UAAtHP,KAAwIQ,GAAA,KAAAN,EAAA,oBAEjJO,oBCCjB,IAuBAC,EAvBAC,EAAA,OAcAC,CACAhB,EACAC,GATA,EAVA,SAAAgB,GACAF,EAAA,SAaA,KAEA,MAUA,oBCvBAG,GADiBhB,OAFjB,WAA0B,IAAaC,EAAbC,KAAaC,eAA0BC,EAAvCF,KAAuCG,MAAAD,IAAAH,EAAwB,OAAAG,EAAA,OAAiBa,YAAA,UAAoBb,EAAA,MAApGF,KAAoGQ,GAApGR,KAAoGgB,GAApGhB,KAAoGiB,WAE7GR,oBCCjB,IAuBAS,EAvBAP,EAAA,OAcAQ,OCRA,wCAIA,iBDMAL,GATA,EAVA,SAAAD,GACAF,EAAA,SAaA,kBAEA,MAUA,QEtBAS,EAAA,EAAIC,IAAIC,EAAA,GAER,IAAAC,EAAA,IAAmBD,EAAA,GACjBE,SAEIC,KAAM,IACNC,KAAM,aACNC,UAAWT,MCLjBE,EAAA,EAAIQ,OAAOC,eAAgB,EAG3B,IAAIT,EAAA,GACFU,GAAI,OACJP,SACAQ,YAAcnC,IAAAc,GACdsB,SAAU","file":"static/js/index.3675151582d13b8e0c7e.js","sourcesContent":["module.exports = \"\"\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/global/assets/logo.png\n// module id = akkm\n// module chunks = 1","\n\n\n\n\n\n\n\n// WEBPACK FOOTER //\n// src/admin/pages/index/App.vue","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',{attrs:{\"id\":\"app\"}},[_c('img',{attrs:{\"src\":_vm.imgurl}}),_vm._v(\" \"),_c('router-view')],1)}\nvar staticRenderFns = []\nvar esExports = { render: render, staticRenderFns: staticRenderFns }\nexport default esExports\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./node_modules/vue-loader/lib/template-compiler?{\"id\":\"data-v-462acfc8\",\"hasScoped\":false,\"transformToRequire\":{\"video\":[\"src\",\"poster\"],\"source\":\"src\",\"img\":\"src\",\"image\":\"xlink:href\"},\"buble\":{\"transforms\":{}}}!./node_modules/vue-loader/lib/selector.js?type=template&index=0!./src/admin/pages/index/App.vue\n// module id = null\n// module chunks = ","function injectStyle (ssrContext) {\n require(\"!!../../../../node_modules/extract-text-webpack-plugin/dist/loader.js?{\\\"omit\\\":1,\\\"remove\\\":true}!vue-style-loader!css-loader?{\\\"sourceMap\\\":true}!../../../../node_modules/vue-loader/lib/style-compiler/index?{\\\"vue\\\":true,\\\"id\\\":\\\"data-v-462acfc8\\\",\\\"scoped\\\":false,\\\"hasInlineConfig\\\":false}!../../../../node_modules/vue-loader/lib/selector?type=styles&index=0!./App.vue\")\n}\nvar normalizeComponent = require(\"!../../../../node_modules/vue-loader/lib/component-normalizer\")\n/* script */\nexport * from \"!!babel-loader!../../../../node_modules/vue-loader/lib/selector?type=script&index=0!./App.vue\"\nimport __vue_script__ from \"!!babel-loader!../../../../node_modules/vue-loader/lib/selector?type=script&index=0!./App.vue\"\n/* template */\nimport __vue_template__ from \"!!../../../../node_modules/vue-loader/lib/template-compiler/index?{\\\"id\\\":\\\"data-v-462acfc8\\\",\\\"hasScoped\\\":false,\\\"transformToRequire\\\":{\\\"video\\\":[\\\"src\\\",\\\"poster\\\"],\\\"source\\\":\\\"src\\\",\\\"img\\\":\\\"src\\\",\\\"image\\\":\\\"xlink:href\\\"},\\\"buble\\\":{\\\"transforms\\\":{}}}!../../../../node_modules/vue-loader/lib/selector?type=template&index=0!./App.vue\"\n/* template functional */\nvar __vue_template_functional__ = false\n/* styles */\nvar __vue_styles__ = injectStyle\n/* scopeId */\nvar __vue_scopeId__ = null\n/* moduleIdentifier (server only) */\nvar __vue_module_identifier__ = null\nvar Component = normalizeComponent(\n __vue_script__,\n __vue_template__,\n __vue_template_functional__,\n __vue_styles__,\n __vue_scopeId__,\n __vue_module_identifier__\n)\n\nexport default Component.exports\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/admin/pages/index/App.vue\n// module id = null\n// module chunks = ","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',{staticClass:\"hello\"},[_c('h1',[_vm._v(_vm._s(_vm.msg))])])}\nvar staticRenderFns = []\nvar esExports = { render: render, staticRenderFns: staticRenderFns }\nexport default esExports\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./node_modules/vue-loader/lib/template-compiler?{\"id\":\"data-v-29427ef6\",\"hasScoped\":true,\"transformToRequire\":{\"video\":[\"src\",\"poster\"],\"source\":\"src\",\"img\":\"src\",\"image\":\"xlink:href\"},\"buble\":{\"transforms\":{}}}!./node_modules/vue-loader/lib/selector.js?type=template&index=0!./src/admin/components/HelloWorld.vue\n// module id = null\n// module chunks = ","function injectStyle (ssrContext) {\n require(\"!!../../../node_modules/extract-text-webpack-plugin/dist/loader.js?{\\\"omit\\\":1,\\\"remove\\\":true}!vue-style-loader!css-loader?{\\\"sourceMap\\\":true}!../../../node_modules/vue-loader/lib/style-compiler/index?{\\\"vue\\\":true,\\\"id\\\":\\\"data-v-29427ef6\\\",\\\"scoped\\\":true,\\\"hasInlineConfig\\\":false}!../../../node_modules/vue-loader/lib/selector?type=styles&index=0!./HelloWorld.vue\")\n}\nvar normalizeComponent = require(\"!../../../node_modules/vue-loader/lib/component-normalizer\")\n/* script */\nexport * from \"!!babel-loader!../../../node_modules/vue-loader/lib/selector?type=script&index=0!./HelloWorld.vue\"\nimport __vue_script__ from \"!!babel-loader!../../../node_modules/vue-loader/lib/selector?type=script&index=0!./HelloWorld.vue\"\n/* template */\nimport __vue_template__ from \"!!../../../node_modules/vue-loader/lib/template-compiler/index?{\\\"id\\\":\\\"data-v-29427ef6\\\",\\\"hasScoped\\\":true,\\\"transformToRequire\\\":{\\\"video\\\":[\\\"src\\\",\\\"poster\\\"],\\\"source\\\":\\\"src\\\",\\\"img\\\":\\\"src\\\",\\\"image\\\":\\\"xlink:href\\\"},\\\"buble\\\":{\\\"transforms\\\":{}}}!../../../node_modules/vue-loader/lib/selector?type=template&index=0!./HelloWorld.vue\"\n/* template functional */\nvar __vue_template_functional__ = false\n/* styles */\nvar __vue_styles__ = injectStyle\n/* scopeId */\nvar __vue_scopeId__ = \"data-v-29427ef6\"\n/* moduleIdentifier (server only) */\nvar __vue_module_identifier__ = null\nvar Component = normalizeComponent(\n __vue_script__,\n __vue_template__,\n __vue_template_functional__,\n __vue_styles__,\n __vue_scopeId__,\n __vue_module_identifier__\n)\n\nexport default Component.exports\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/admin/components/HelloWorld.vue\n// module id = null\n// module chunks = ","\n\n\n\n\n\n\n\n\n// WEBPACK FOOTER //\n// src/admin/components/HelloWorld.vue","import Vue from 'vue'\nimport Router from 'vue-router'\nimport HelloWorld from '@/components/HelloWorld'\n\nVue.use(Router)\n\nexport default new Router({\n routes: [\n {\n path: '/',\n name: 'HelloWorld',\n component: HelloWorld\n }\n ]\n})\n\n\n\n// WEBPACK FOOTER //\n// ./src/admin/router/index.js","// The Vue build version to load with the `import` command\n// (runtime-only or standalone) has been set in webpack.base.conf with an alias.\nimport Vue from 'vue'\nimport App from './App'\nimport router from '../../router';\n\nVue.config.productionTip = false\n\n/* eslint-disable no-new */\nnew Vue({\n el: '#app',\n router,\n components: { App },\n template: ''\n})\n\n\n\n// WEBPACK FOOTER //\n// ./src/admin/pages/index/index.js"],"sourceRoot":""} -------------------------------------------------------------------------------- /dist/admin/static/js/manifest.2ae2e69a05c33dfc65f8.js: -------------------------------------------------------------------------------- 1 | !function(r){var n=window.webpackJsonp;window.webpackJsonp=function(e,u,c){for(var f,i,p,a=0,l=[];avue-multi-project
-------------------------------------------------------------------------------- /dist/master/static/css/index.7cb8a531774c9910737734e1c15db140.css: -------------------------------------------------------------------------------- 1 | #app{font-family:Avenir,Helvetica,Arial,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;text-align:center;color:#2c3e50;margin-top:60px}h1[data-v-4208fb8b],h2[data-v-4208fb8b]{font-weight:400}ul[data-v-4208fb8b]{list-style-type:none;padding:0}li[data-v-4208fb8b]{display:inline-block;margin:0 10px}a[data-v-4208fb8b]{color:#42b983} 2 | /*# sourceMappingURL=index.7cb8a531774c9910737734e1c15db140.css.map */ -------------------------------------------------------------------------------- /dist/master/static/css/index.7cb8a531774c9910737734e1c15db140.css.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["index.7cb8a531774c9910737734e1c15db140.css"],"names":[],"mappings":"AACA,KACE,8CAAoD,AACpD,mCAAoC,AACpC,kCAAmC,AACnC,kBAAmB,AACnB,cAAe,AACf,eAAiB,CAClB,AAED,wCACE,eAAoB,CACrB,AACD,oBACE,qBAAsB,AACtB,SAAW,CACZ,AACD,oBACE,qBAAsB,AACtB,aAAe,CAChB,AACD,mBACE,aAAe,CAChB","file":"index.7cb8a531774c9910737734e1c15db140.css","sourcesContent":["\n#app {\n font-family: 'Avenir', Helvetica, Arial, sans-serif;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n text-align: center;\n color: #2c3e50;\n margin-top: 60px;\n}\n\nh1[data-v-4208fb8b], h2[data-v-4208fb8b] {\n font-weight: normal;\n}\nul[data-v-4208fb8b] {\n list-style-type: none;\n padding: 0;\n}\nli[data-v-4208fb8b] {\n display: inline-block;\n margin: 0 10px;\n}\na[data-v-4208fb8b] {\n color: #42b983;\n}\n"]} -------------------------------------------------------------------------------- /dist/master/static/js/index.0f86190d29b252e0535a.js: -------------------------------------------------------------------------------- 1 | webpackJsonp([1],{5863:function(t,i){},akkm:function(t,i){t.exports=""},"u/vV":function(t,i,c){"use strict";Object.defineProperty(i,"__esModule",{value:!0});var I=c("7+uW"),n={name:"App",data:function(){return{imgurl:c("akkm")}}},l={render:function(){var t=this.$createElement,i=this._self._c||t;return i("div",{attrs:{id:"app"}},[i("img",{attrs:{src:this.imgurl}}),this._v(" "),i("router-view")],1)},staticRenderFns:[]};var s=c("VU/8")(n,l,!1,function(t){c("xV2Q")},null,null).exports,e=c("/ocq"),a={render:function(){var t=this.$createElement,i=this._self._c||t;return i("div",{staticClass:"hello"},[i("h1",[this._v(this._s(this.msg))])])},staticRenderFns:[]};var h=c("VU/8")({name:"HelloWorld",data:function(){return{msg:"欢迎来到master项目"}}},a,!1,function(t){c("5863")},"data-v-4208fb8b",null).exports;I.a.use(e.a);var R=new e.a({routes:[{path:"/",name:"HelloWorld",component:h}]});I.a.config.productionTip=!1,new I.a({el:"#app",router:R,components:{App:s},template:""})},xV2Q:function(t,i){}},["u/vV"]); 2 | //# sourceMappingURL=index.0f86190d29b252e0535a.js.map -------------------------------------------------------------------------------- /dist/master/static/js/index.0f86190d29b252e0535a.js.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["webpack:///./src/global/assets/logo.png","webpack:///src/master/pages/index/App.vue","webpack:///./src/master/pages/index/App.vue?12fc","webpack:///./src/master/pages/index/App.vue","webpack:///./src/master/components/HelloWorld.vue?d6be","webpack:///./src/master/components/HelloWorld.vue","webpack:///src/master/components/HelloWorld.vue","webpack:///./src/master/router/index.js","webpack:///./src/master/pages/index/index.js"],"names":["module","exports","App","index_App","render","_h","this","$createElement","_c","_self","attrs","id","src","imgurl","_v","staticRenderFns","pages_index_App","__webpack_require__","normalizeComponent","ssrContext","components_HelloWorld","staticClass","_s","msg","master_components_HelloWorld","HelloWorld_normalizeComponent","vue_esm","use","vue_router_esm","router","routes","path","name","component","config","productionTip","el","components","template"],"mappings":"0DAAAA,EAAAC,QAAA,2iSCQAC,QAEA,sCAIA,WCXAC,GADiBC,OAFjB,WAA0B,IAAaC,EAAbC,KAAaC,eAA0BC,EAAvCF,KAAuCG,MAAAD,IAAAH,EAAwB,OAAAG,EAAA,OAAiBE,OAAOC,GAAA,SAAYH,EAAA,OAAYE,OAAOE,IAAtHN,KAAsHO,UAAtHP,KAAwIQ,GAAA,KAAAN,EAAA,oBAEjJO,oBCCjB,IAuBAC,EAvBAC,EAAA,OAcAC,CACAhB,EACAC,GATA,EAVA,SAAAgB,GACAF,EAAA,SAaA,KAEA,MAUA,oBCvBAG,GADiBhB,OAFjB,WAA0B,IAAaC,EAAbC,KAAaC,eAA0BC,EAAvCF,KAAuCG,MAAAD,IAAAH,EAAwB,OAAAG,EAAA,OAAiBa,YAAA,UAAoBb,EAAA,MAApGF,KAAoGQ,GAApGR,KAAoGgB,GAApGhB,KAAoGiB,WAE7GR,oBCCjB,IAuBAS,EAvBAP,EAAA,OAcAQ,OCRA,wCAIA,kBDMAL,GATA,EAVA,SAAAD,GACAF,EAAA,SAaA,kBAEA,MAUA,QEtBAS,EAAA,EAAIC,IAAIC,EAAA,GAER,IAAAC,EAAA,IAAmBD,EAAA,GACjBE,SAEIC,KAAM,IACNC,KAAM,aACNC,UAAWT,MCLjBE,EAAA,EAAIQ,OAAOC,eAAgB,EAG3B,IAAIT,EAAA,GACFU,GAAI,OACJP,SACAQ,YAAcnC,IAAAc,GACdsB,SAAU","file":"static/js/index.0f86190d29b252e0535a.js","sourcesContent":["module.exports = \"\"\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/global/assets/logo.png\n// module id = akkm\n// module chunks = 1","\n\n\n\n\n\n\n\n// WEBPACK FOOTER //\n// src/master/pages/index/App.vue","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',{attrs:{\"id\":\"app\"}},[_c('img',{attrs:{\"src\":_vm.imgurl}}),_vm._v(\" \"),_c('router-view')],1)}\nvar staticRenderFns = []\nvar esExports = { render: render, staticRenderFns: staticRenderFns }\nexport default esExports\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./node_modules/vue-loader/lib/template-compiler?{\"id\":\"data-v-66b06aea\",\"hasScoped\":false,\"transformToRequire\":{\"video\":[\"src\",\"poster\"],\"source\":\"src\",\"img\":\"src\",\"image\":\"xlink:href\"},\"buble\":{\"transforms\":{}}}!./node_modules/vue-loader/lib/selector.js?type=template&index=0!./src/master/pages/index/App.vue\n// module id = null\n// module chunks = ","function injectStyle (ssrContext) {\n require(\"!!../../../../node_modules/extract-text-webpack-plugin/dist/loader.js?{\\\"omit\\\":1,\\\"remove\\\":true}!vue-style-loader!css-loader?{\\\"sourceMap\\\":true}!../../../../node_modules/vue-loader/lib/style-compiler/index?{\\\"vue\\\":true,\\\"id\\\":\\\"data-v-66b06aea\\\",\\\"scoped\\\":false,\\\"hasInlineConfig\\\":false}!../../../../node_modules/vue-loader/lib/selector?type=styles&index=0!./App.vue\")\n}\nvar normalizeComponent = require(\"!../../../../node_modules/vue-loader/lib/component-normalizer\")\n/* script */\nexport * from \"!!babel-loader!../../../../node_modules/vue-loader/lib/selector?type=script&index=0!./App.vue\"\nimport __vue_script__ from \"!!babel-loader!../../../../node_modules/vue-loader/lib/selector?type=script&index=0!./App.vue\"\n/* template */\nimport __vue_template__ from \"!!../../../../node_modules/vue-loader/lib/template-compiler/index?{\\\"id\\\":\\\"data-v-66b06aea\\\",\\\"hasScoped\\\":false,\\\"transformToRequire\\\":{\\\"video\\\":[\\\"src\\\",\\\"poster\\\"],\\\"source\\\":\\\"src\\\",\\\"img\\\":\\\"src\\\",\\\"image\\\":\\\"xlink:href\\\"},\\\"buble\\\":{\\\"transforms\\\":{}}}!../../../../node_modules/vue-loader/lib/selector?type=template&index=0!./App.vue\"\n/* template functional */\nvar __vue_template_functional__ = false\n/* styles */\nvar __vue_styles__ = injectStyle\n/* scopeId */\nvar __vue_scopeId__ = null\n/* moduleIdentifier (server only) */\nvar __vue_module_identifier__ = null\nvar Component = normalizeComponent(\n __vue_script__,\n __vue_template__,\n __vue_template_functional__,\n __vue_styles__,\n __vue_scopeId__,\n __vue_module_identifier__\n)\n\nexport default Component.exports\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/master/pages/index/App.vue\n// module id = null\n// module chunks = ","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',{staticClass:\"hello\"},[_c('h1',[_vm._v(_vm._s(_vm.msg))])])}\nvar staticRenderFns = []\nvar esExports = { render: render, staticRenderFns: staticRenderFns }\nexport default esExports\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./node_modules/vue-loader/lib/template-compiler?{\"id\":\"data-v-4208fb8b\",\"hasScoped\":true,\"transformToRequire\":{\"video\":[\"src\",\"poster\"],\"source\":\"src\",\"img\":\"src\",\"image\":\"xlink:href\"},\"buble\":{\"transforms\":{}}}!./node_modules/vue-loader/lib/selector.js?type=template&index=0!./src/master/components/HelloWorld.vue\n// module id = null\n// module chunks = ","function injectStyle (ssrContext) {\n require(\"!!../../../node_modules/extract-text-webpack-plugin/dist/loader.js?{\\\"omit\\\":1,\\\"remove\\\":true}!vue-style-loader!css-loader?{\\\"sourceMap\\\":true}!../../../node_modules/vue-loader/lib/style-compiler/index?{\\\"vue\\\":true,\\\"id\\\":\\\"data-v-4208fb8b\\\",\\\"scoped\\\":true,\\\"hasInlineConfig\\\":false}!../../../node_modules/vue-loader/lib/selector?type=styles&index=0!./HelloWorld.vue\")\n}\nvar normalizeComponent = require(\"!../../../node_modules/vue-loader/lib/component-normalizer\")\n/* script */\nexport * from \"!!babel-loader!../../../node_modules/vue-loader/lib/selector?type=script&index=0!./HelloWorld.vue\"\nimport __vue_script__ from \"!!babel-loader!../../../node_modules/vue-loader/lib/selector?type=script&index=0!./HelloWorld.vue\"\n/* template */\nimport __vue_template__ from \"!!../../../node_modules/vue-loader/lib/template-compiler/index?{\\\"id\\\":\\\"data-v-4208fb8b\\\",\\\"hasScoped\\\":true,\\\"transformToRequire\\\":{\\\"video\\\":[\\\"src\\\",\\\"poster\\\"],\\\"source\\\":\\\"src\\\",\\\"img\\\":\\\"src\\\",\\\"image\\\":\\\"xlink:href\\\"},\\\"buble\\\":{\\\"transforms\\\":{}}}!../../../node_modules/vue-loader/lib/selector?type=template&index=0!./HelloWorld.vue\"\n/* template functional */\nvar __vue_template_functional__ = false\n/* styles */\nvar __vue_styles__ = injectStyle\n/* scopeId */\nvar __vue_scopeId__ = \"data-v-4208fb8b\"\n/* moduleIdentifier (server only) */\nvar __vue_module_identifier__ = null\nvar Component = normalizeComponent(\n __vue_script__,\n __vue_template__,\n __vue_template_functional__,\n __vue_styles__,\n __vue_scopeId__,\n __vue_module_identifier__\n)\n\nexport default Component.exports\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./src/master/components/HelloWorld.vue\n// module id = null\n// module chunks = ","\n\n\n\n\n\n\n\n\n// WEBPACK FOOTER //\n// src/master/components/HelloWorld.vue","import Vue from 'vue'\nimport Router from 'vue-router'\nimport HelloWorld from '@/components/HelloWorld'\n\nVue.use(Router)\n\nexport default new Router({\n routes: [\n {\n path: '/',\n name: 'HelloWorld',\n component: HelloWorld\n }\n ]\n})\n\n\n\n// WEBPACK FOOTER //\n// ./src/master/router/index.js","// The Vue build version to load with the `import` command\n// (runtime-only or standalone) has been set in webpack.base.conf with an alias.\nimport Vue from 'vue'\nimport App from './App'\nimport router from '../../router';\n\nVue.config.productionTip = false\n\n/* eslint-disable no-new */\nnew Vue({\n el: '#app',\n router,\n components: { App },\n template: ''\n})\n\n\n\n// WEBPACK FOOTER //\n// ./src/master/pages/index/index.js"],"sourceRoot":""} -------------------------------------------------------------------------------- /dist/master/static/js/manifest.2ae2e69a05c33dfc65f8.js: -------------------------------------------------------------------------------- 1 | !function(r){var n=window.webpackJsonp;window.webpackJsonp=function(e,u,c){for(var f,i,p,a=0,l=[];a= 6.0.0", 63 | "npm": ">= 3.0.0" 64 | }, 65 | "browserslist": [ 66 | "> 1%", 67 | "last 2 versions", 68 | "not ie <= 8" 69 | ] 70 | } 71 | -------------------------------------------------------------------------------- /src/admin/components/HelloWorld.vue: -------------------------------------------------------------------------------- 1 | 6 | 7 | 17 | 18 | 19 | 35 | -------------------------------------------------------------------------------- /src/admin/config/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | module.exports = { 3 | name : 'admin', //文件夹名称 和最后打包后文件夹的名称 /src/admin/ 4 | } -------------------------------------------------------------------------------- /src/admin/pages/index/App.vue: -------------------------------------------------------------------------------- 1 | 7 | 8 | 18 | 19 | 29 | -------------------------------------------------------------------------------- /src/admin/pages/index/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/augustVino/vue-multi-project/05f96068f4ab59aeceaff6ca187a0fdd3587dfad/src/admin/pages/index/favicon.ico -------------------------------------------------------------------------------- /src/admin/pages/index/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | vue-multi-project 7 | 8 | 9 |
10 | 11 | 12 | -------------------------------------------------------------------------------- /src/admin/pages/index/index.js: -------------------------------------------------------------------------------- 1 | // The Vue build version to load with the `import` command 2 | // (runtime-only or standalone) has been set in webpack.base.conf with an alias. 3 | import Vue from 'vue' 4 | import App from './App' 5 | import router from '../../router'; 6 | 7 | Vue.config.productionTip = false 8 | 9 | /* eslint-disable no-new */ 10 | new Vue({ 11 | el: '#app', 12 | router, 13 | components: { App }, 14 | template: '' 15 | }) 16 | -------------------------------------------------------------------------------- /src/admin/router/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import Router from 'vue-router' 3 | import HelloWorld from '@/components/HelloWorld' 4 | 5 | Vue.use(Router) 6 | 7 | export default new Router({ 8 | routes: [ 9 | { 10 | path: '/', 11 | name: 'HelloWorld', 12 | component: HelloWorld 13 | } 14 | ] 15 | }) 16 | -------------------------------------------------------------------------------- /src/global/assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/augustVino/vue-multi-project/05f96068f4ab59aeceaff6ca187a0fdd3587dfad/src/global/assets/logo.png -------------------------------------------------------------------------------- /src/master/components/HelloWorld.vue: -------------------------------------------------------------------------------- 1 | 6 | 7 | 17 | 18 | 19 | 35 | -------------------------------------------------------------------------------- /src/master/config/index.js: -------------------------------------------------------------------------------- 1 | 'use strict' 2 | module.exports = { 3 | name : 'master', //文件夹名称 和最后打包后文件夹的名称 /src/admin/ 4 | } -------------------------------------------------------------------------------- /src/master/pages/index/App.vue: -------------------------------------------------------------------------------- 1 | 7 | 8 | 18 | 19 | 29 | -------------------------------------------------------------------------------- /src/master/pages/index/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/augustVino/vue-multi-project/05f96068f4ab59aeceaff6ca187a0fdd3587dfad/src/master/pages/index/favicon.ico -------------------------------------------------------------------------------- /src/master/pages/index/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | vue-multi-project 7 | 8 | 9 |
10 | 11 | 12 | -------------------------------------------------------------------------------- /src/master/pages/index/index.js: -------------------------------------------------------------------------------- 1 | // The Vue build version to load with the `import` command 2 | // (runtime-only or standalone) has been set in webpack.base.conf with an alias. 3 | import Vue from 'vue' 4 | import App from './App' 5 | import router from '../../router'; 6 | 7 | Vue.config.productionTip = false 8 | 9 | /* eslint-disable no-new */ 10 | new Vue({ 11 | el: '#app', 12 | router, 13 | components: { App }, 14 | template: '' 15 | }) 16 | -------------------------------------------------------------------------------- /src/master/router/index.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import Router from 'vue-router' 3 | import HelloWorld from '@/components/HelloWorld' 4 | 5 | Vue.use(Router) 6 | 7 | export default new Router({ 8 | routes: [ 9 | { 10 | path: '/', 11 | name: 'HelloWorld', 12 | component: HelloWorld 13 | } 14 | ] 15 | }) 16 | -------------------------------------------------------------------------------- /static/.gitkeep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/augustVino/vue-multi-project/05f96068f4ab59aeceaff6ca187a0fdd3587dfad/static/.gitkeep -------------------------------------------------------------------------------- /test/unit/.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "jest": true 4 | }, 5 | "globals": { 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /test/unit/jest.conf.js: -------------------------------------------------------------------------------- 1 | const path = require('path') 2 | 3 | module.exports = { 4 | rootDir: path.resolve(__dirname, '../../'), 5 | moduleFileExtensions: [ 6 | 'js', 7 | 'json', 8 | 'vue' 9 | ], 10 | moduleNameMapper: { 11 | '^@/(.*)$': '/src/$1' 12 | }, 13 | transform: { 14 | '^.+\\.js$': '/node_modules/babel-jest', 15 | '.*\\.(vue)$': '/node_modules/vue-jest' 16 | }, 17 | snapshotSerializers: ['/node_modules/jest-serializer-vue'], 18 | setupFiles: ['/test/unit/setup'], 19 | mapCoverage: true, 20 | coverageDirectory: '/test/unit/coverage', 21 | collectCoverageFrom: [ 22 | 'src/**/*.{js,vue}', 23 | '!src/main.js', 24 | '!src/router/index.js', 25 | '!**/node_modules/**' 26 | ] 27 | } 28 | -------------------------------------------------------------------------------- /test/unit/setup.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | 3 | Vue.config.productionTip = false 4 | -------------------------------------------------------------------------------- /test/unit/specs/HelloWorld.spec.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | import HelloWorld from '@/components/HelloWorld' 3 | 4 | describe('HelloWorld.vue', () => { 5 | it('should render correct contents', () => { 6 | const Constructor = Vue.extend(HelloWorld) 7 | const vm = new Constructor().$mount() 8 | expect(vm.$el.querySelector('.hello h1').textContent) 9 | .toEqual('Welcome to Your Vue.js App') 10 | }) 11 | }) 12 | --------------------------------------------------------------------------------