美文网首页react+webpack
使用 webpack3 配置多页应用(四)

使用 webpack3 配置多页应用(四)

作者: 柏丘君 | 来源:发表于2017-09-11 16:21 被阅读1801次

    本文将提供我项目中一份完整的配置文件,供大家参考。

    package.json

    {
      "name": "xxx",
      "version": "1.0.0",
      "description": "",
      "main": "index.js",
      "scripts": {
        "dev": "set NODE_ENV=dev && webpack-dev-server --open",
        "build": "set NODE_ENV=prod && webpack -p",
        "lint": "set NODE_ENV=lint && webpack-dev-server --open",
        "serve": "http-server ./dist -p 8888 -o",
        "serve2": "http-server ./dist -p 8888"
      },
      "author": "",
      "license": "ISC",
      "devDependencies": {
        "autoprefixer": "^7.1.3",
        "babel-core": "^6.26.0",
        "babel-loader": "^7.1.2",
        "babel-plugin-transform-es2015-spread": "^6.22.0",
        "babel-preset-env": "^1.6.0",
        "clean-webpack-plugin": "^0.1.16",
        "css-loader": "^0.28.7",
        "eslint": "^4.5.0",
        "eslint-loader": "^1.9.0",
        "extract-text-webpack-plugin": "^3.0.0",
        "file-loader": "^0.11.2",
        "html-webpack-plugin": "^2.30.1",
        "http-server": "^0.10.0",
        "postcss-loader": "^2.0.6",
        "style-loader": "^0.18.2",
        "url-loader": "^0.5.9",
        "webpack": "^3.5.5",
        "webpack-dev-server": "^2.7.1",
        "webpack-merge": "^4.1.0"
      },
      "dependencies": {}
    }
    

    .gitignore

    node_modules
    dist
    npm-debug.log
    

    .babelrc

    {
        "plugins": ["transform-es2015-spread"]
    }
    

    .eslintrc.js

    module.exports = {
      env: {
        browser: true,
        commonjs: true,
        es6: true,
        node: true,
      },
      extends: 'eslint:recommended',
      parserOptions: {
        sourceType: 'module',
      },
      rules: {
        'comma-dangle': ['error', 'always-multiline'],
        indent: ['error', 2],
        'linebreak-style': ['error', 'unix'],
        quotes: ['error', 'single'],
        semi: ['error', 'always'],
        'no-unused-vars': ['warn'],
        'no-console': 0,
      },
    };
    

    postcss.config.js

    module.exports = {  
      plugins: {  
        'autoprefixer': {
            browsers: ['last 5 version','Android >= 4.0'],
            //是否美化属性值 默认:true 
            cascade: true,
            //是否去掉不必要的前缀 默认:true 
            remove: true
        }  
      }  
    }  
    

    config.js

    module.exports = {
        HTMLDirs:[
            "index",
            "company_intro",
            "enterprise_culture",
            "hornors",
            "news_center",
            "news_item",
            "product",
            "schools",
            "operate",
            "cooperate",
            "join_us",
            "contact_us",
            "investment"
        ],
        cssPublicPath:"../",
        imgOutputPath:"img/",
        cssOutputPath:"./css/styles.css",
        devServerOutputPath:"../dist",
    
    }
    

    webpack.config.js

    // 获取环境命令,并去除首尾空格
    const env = process.env.NODE_ENV.replace(/(\s*$)|(^\s*)/ig,"");
    // 根据环境变量引用相关的配置文件
    module.exports = require(`./config/webpack.config.${env}.js`)
    

    webpack.config.base.js

    const path = require("path");
    // 引入插件
    const HTMLWebpackPlugin = require("html-webpack-plugin");
    // 清理 dist 文件夹
    const CleanWebpackPlugin = require("clean-webpack-plugin")
    // 抽取 css
    const ExtractTextPlugin = require("extract-text-webpack-plugin");
    // 引入多页面文件列表
    const config = require("./config");
    // 通过 html-webpack-plugin 生成的 HTML 集合
    let HTMLPlugins = [];
    // 入口文件集合
    let Entries = {}
    
    // 生成多页面的集合
    config.HTMLDirs.forEach((page) => {
        const htmlPlugin = new HTMLWebpackPlugin({
            filename: `${page}.html`,
            template: path.resolve(__dirname, `../app/html/${page}.html`),
            chunks: [page, 'commons'],
        });
        HTMLPlugins.push(htmlPlugin);
        Entries[page] = path.resolve(__dirname, `../app/js/${page}.js`);
    })
    
    module.exports = {
        entry:Entries,
        devtool:"cheap-module-source-map",
        output:{
            filename:"js/[name].bundle.[hash].js",
            path:path.resolve(__dirname,"../dist")
        },
        // 加载器
        module:{
            rules:[
                {
                    // 对 css 后缀名进行处理
                    test:/\.css$/,
                    // 不处理 node_modules 文件中的 css 文件
                    exclude: /node_modules/,
                    // 抽取 css 文件到单独的文件夹
                    use: ExtractTextPlugin.extract({
                        fallback: "style-loader",
                        // 设置 css 的 publicPath
                        publicPath: config.cssPublicPath,
                        use: [{
                                loader:"css-loader",
                                options:{
                                    // 开启 css 压缩
                                    minimize:true,
                                }
                            },
                            {
                                loader:"postcss-loader",
                            }
                        ]
                    })
                },
                {
                    test: /\.js$/,
                    exclude: /node_modules/,
                    use: {
                        loader: 'babel-loader',
                        options: {
                            presets: ['env']
                        }
                    }
                },
                {
                    test: /\.(png|svg|jpg|gif)$/,
                    use:{
                        loader:"file-loader",
                        options:{
                            // 打包生成图片的名字
                            name:"[name].[ext]",
                            // 图片的生成路径
                            outputPath:config.imgOutputPath
                        }
                    }
                },
                {
                    test: /\.(woff|woff2|eot|ttf|otf)$/,
                    use:["file-loader"]
                }
            ],
        },
        plugins:[
            // 自动清理 dist 文件夹
            new CleanWebpackPlugin(["dist"]),
            // 将 css 抽取到某个文件夹
            new ExtractTextPlugin(config.cssOutputPath),        
            // 自动生成 HTML 插件
            ...HTMLPlugins
        ],
    }
    

    webpack.config.dev.js

    // 引入基础配置文件
    const webpackBase = require("./webpack.config.base");
    // 引入 webpack-merge 插件
    const webpackMerge = require("webpack-merge");
    // 引入配置文件
    const config = require("./config");
    // 合并配置文件
    module.exports = webpackMerge(webpackBase,{
        // 配置 webpack-dev-server
        devServer:{
            // 项目根目录
            contentBase:config.devServerOutputPath,
            // 错误、警告展示设置
            overlay:{
                errors:true,
                warnings:true
            }
        }
    });
    

    webpack.config.prod.js

    // 引入基础配置
    const webpackBase = require("./webpack.config.base");
    // 引入 webpack-merge 插件
    const webpackMerge = require("webpack-merge");
    // 引入 webpack
    const webpack = require("webpack");
    // 合并配置文件
    module.exports = webpackMerge(webpackBase,{
        plugins:[
            // 代码压缩
            new webpack.optimize.UglifyJsPlugin({
                // 开启 sourceMap
                sourceMap: true
            }),
            // 提取公共 JavaScript 代码
            new webpack.optimize.CommonsChunkPlugin({
                // chunk 名为 commons
                name: "commons",
                filename: "[name].bundle.js",
            }),
        ]
    });
    

    webpack.config.lint.js

    const webpackBase = require("./webpack.config.base");
    const webpackMerge = require("webpack-merge");
    const config = require("./config");
    module.exports = webpackMerge(webpackBase,{
        module:{
            rules:[
                {
                    test: /\.js$/,
                    // 强制先进行 ESLint 检查
                    enforce: "pre",
                    // 不对 node_modules 和 lib 文件夹中的代码进行检查
                    exclude: /node_modules|lib/,
                    loader: "eslint-loader",
                    options: {
                        // 启用自动修复
                        fix:true,
                        // 启用警告信息
                        emitWarning:true,
                    }
                },
            ]
        },
        devServer:{
            contentBase:config.devServerOutputPath,
            overlay:{
                errors:true,
                warnings:true
            }
        }
    });
    

    项目结构

    │  .babelrc
    │  .eslintrc.js
    │  .gitignore
    │  package.json
    │  postcss.config.js
    │  webpack.config.js
    │  
    ├─app
    │  │  favicon.ico
    │  │  
    │  ├─css
    │  │      main.css
    │  │      
    │  ├─html
    │  │      index.html
    │  │    
    │  │      
    │  ├─img
    │  │      back.png
    │  │      
    │  ├─js
    │  │      ajax.js
    │  │      footer.js
    │  │      index.js
    │  │      nav.js
    │  │      public.js
    │  │      tity_nav.js
    │  │      
    │  └─lib
    │        flexible.js
    │        normalize.css
    │        swiper.css
    │        swiper.js
    │        
    └─config
            config.js
            webpack.config.base.js
            webpack.config.dev.js
            webpack.config.lint.js
            webpack.config.prod.js
    

    完。

    相关文章

      网友评论

      • 启航_b057:我的css抽离不出来 有么有大大帮忙解决下 没有报错。。。。
      • 8582f7c67ed7:我重新搭建了一下,其他的都没问题,就是打包时不清除重复的文件,有可能是什么原因导致的?
        前端学习者:按照这个配置的话,是不是没有对html中的img做处理
      • Vusui:请问GIT上面有这个完整代码吗?:+1:
        8582f7c67ed7:我自己写了一个,需要可以分享给你
      • jed_shi:为了方便小白以及偷懒不写所有的文件名,将`config`文件夹下的`config.js`更改为适合本demo的版本。
        ···
        // 获取所有页面 生成多页面的集合
        let fs = require("fs");
        const getFileNameList = path => {
        let fileList = [];
        let dirList = fs.readdirSync(path);
        dirList.forEach(item => {
        if (item.indexOf('html') > -1) {
        fileList.push(item.split('.')[0]);
        }
        });
        return fileList; //返回当前html文件夹下的所有文件名组成的数组
        };

        module.exports = {

        HTMLDirs: getFileNameList('./app/html'),
        cssPublicPath: "../",
        imgOutputPath: "img/",
        cssOutputPath: "./css/styles.css",
        devServerOutputPath: "../dist",

        }
        ···
        感谢评论区 @赵的拇指 :pray:
        柏丘君:漂亮:+1:
      • 呜哇呜哇啦啦啦:大大 我照着你的搞了一遍 发现图片没有按套路打包进dist目录啊 这个是为啥子咧
        eee178140ade:@呜哇呜哇啦啦啦 问一下你那个图片怎么打包到dist目录里,我得怎么都打不进去
        呜哇呜哇啦啦啦:@黑黢黢 解决了,使用html-loader就好了,谢谢大大的回复
        柏丘君:@呜哇呜哇啦啦啦 有没有报错信息呢?
      • 銘祈lsvin:学习学习,我在.html文件加个<img />标签发现读取不到,请问这是什么问题
      • louhangfei:大神,我已经传到github上了。
        地址:https://github.com/i3web/webpack-multiple-pages.git
        但是仍然有一个bug,我运行npm run build或npm run dev的时候,会报 Cannot read property 'replace' of undefined的错误。是因为process.env.NODE_ENV为undefined。我用的是mac。 "dev": "cross-env NODE_ENV=dev && webpack-dev-server --open"。能帮我看看吗?
        柏丘君:@louhangfei cross env的官方文档上没有加
        louhangfei:@黑黢黢 卧槽,居然真的可以了。这是为什么呢?加了&&怎么就不行了呢
        柏丘君:@louhangfei 看下这样可以不:cross-env NODE_ENV=dev webpack-dev-server --open
      • louhangfei:提一个建议:webpack.config.base.js中`output:{
        filename:"js/[name].bundle.[hash].js",
        path:path.resolve(__dirname,"../dist")
        },`
        这里的[hash]改为[chunkhash]。这是因为前者每次build时生成的版本号都不一样,即使内容没改动。而后者当内容无改动时,生成的版本号是不变的。有利于做对比。
        参见https://doc.webpack-china.org/guides/caching/
        柏丘君:@louhangfei 谢谢您的建议!
      • louhangfei:现在报了这个错误
        ERROR in chunk index [entry]
        bundle.js
        Conflict: Multiple assets emit to the same filename bundle.js

        ERROR in chunk about [entry]
        bundle.js
        Conflict: Multiple assets emit to the same filename bundle.js
        louhangfei:解决了。webpack.config.base.js没写全
        module.exports = {
        entry: Entries,
        devtool: 'cheap-module-source-map',
        output:{
        filename:"js/[name].bundle.[hash].js",
        path:path.resolve(__dirname,"../dist")
        },
        modules:...
        }
      • louhangfei:写得很棒,建议整理一下放到github上。这份配置文件对于多页面项目的开发很有通用性。
        jed_shi:@黑黢黢 真棒 我给你点赞!
        柏丘君:@louhangfei 可以,这是半年前写的配置,还有很多改进的地方,稍后可以整理下
      • 晒德乌漆麻黑:楼主 请问每个html页面都必须要有对应的js吗?

        ERROR in multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/about.js
        Module not found: Error: Can't resolve 'D:\MyStudys\webpack_blog\app\js\about.js' in 'D:\MyStudys\webpack_blog'
        @ multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/about.js

        ERROR in multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/blog.js
        Module not found: Error: Can't resolve 'D:\MyStudys\webpack_blog\app\js\blog.js' in 'D:\MyStudys\webpack_blog'
        @ multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/blog.js

        ERROR in multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/error.js
        Module not found: Error: Can't resolve 'D:\MyStudys\webpack_blog\app\js\error.js' in 'D:\MyStudys\webpack_blog'
        @ multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/error.js
        晒德乌漆麻黑:@黑黢黢 可以教我怎么修改吗?我是刚接触的小白:joy:

        eg:我有5个html页面 有六个js 有4个css 有的html引用了多个js和css 有的页面没有引用 我要怎么去合并js和css 求指教:blush:
        柏丘君:@晒德乌漆麻黑 我的配置是这样的哈,可以进行修改哦,不必每个页面都有js的
      • 福尔摩猪2333:大大,请问js跟css怎么引入呀?小白一枚,不怎么会用:persevere:
      • Tomotoes:真棒的文章!
        Tomotoes:@黑黢黢 您的配置文件思路给了我很大的启发:blush:
        柏丘君:谢谢你!
      • ZombieBrandg:博主按照您的方法我运行npm run dev报错了!
        const env = process.env.NODE_ENV.replace(/(\s*$)|(^\s*)/);
        TypeError: Cannot read property 'replace' of undefined
        柏丘君:@ZombieBrandg 你用的mac系统吗,可能对环境变量设置不兼容
        ZombieBrandg:@黑黢黢 package.json里这段有设置...
        "scripts": {
        "dev": "set NODE_ENV=dev && webpack-dev-server --open",
        "build": "set NODE_ENV=prod && webpack -p",
        "lint": "set NODE_ENV=lint && webpack-dev-server --open",
        "serve": "http-server ./dist -p 8888 -o",
        "serve2": "http-server ./dist -p 8888"
        },
        柏丘君:@ZombieBrandg 你可能是忘记在package.json设置环境变量了
      • 斜影醉清风丶浪表哥:多页的话css 单独抽离成一个,项目大的话首屏加载会有一定影响的。
      • magic恬:按照您的思路进行一番测试,然而报错了
        ERROR in multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/${page}.js
        Module not found: Error: Can't resolve 'F:\webpackMy\my\morepage\app\js\${page}.js' in 'F:\webpackMy\my\morepage'
        @ multi (webpack)-dev-server/client?http://localhost:8080 ./app/js/${page}.js
        louhangfei:@magic恬 路径要用反引号,不是单引号
      • ae51f1506b38:有报错啊,可以下载完整的代码吗?
        柏丘君:@三行code 清空的不是最外层的dist目录可能是我配置的路径问题,你把路径再向上指一层试试,“还有css目录下的没有编译”是什么意思?你声明的css样式没有生效吗?
        ae51f1506b38:@黑黢黢 错误调好了。dist文件夹没有自动清空,提示是清空了‘config/dist’目录,不是外层的,还有css目录下的没有编译,求指点迷津
        柏丘君:@三行code 请问报什么错误?
      • 659e1bad3a65:请教下有没有插件可以完成一个html里引入其他html的功能,在gulp里有gulp-file-include可以这样引入文件@@include('../../component/header/index.inc'),用webpack的找了很久一直没找到合适的
        柏丘君: @659e1bad3a65 我当初也有这样的需求,不过没有找插件。我是这些元件集成到一个类里面,比如Header类,使用ES6的模板字符串拼接HTML,这样就够成了一个个的class,然后在需要的页面引入这些class并new一下。这是我的做法哈,不知对你有没有帮助
        659e1bad3a65:@黑黢黢 对的
        柏丘君: @659e1bad3a65 你是说分模块吗?比如将header和footer放在一个文件夹,然后在页面中引入?
      • loveoflife_7655:尽然报错了
        loveoflife_7655:@黑黢黢 :+1: 这篇文章不错欧。已经配置成功
        loveoflife_7655:@黑黢黢 搞定了。建议html模版那块改成自动获取文件名
        柏丘君:咦,哪里报错?

      本文标题:使用 webpack3 配置多页应用(四)

      本文链接:https://www.haomeiwen.com/subject/bhsfsxtx.html