webpack.config.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  1. const HtmlWebpackPlugin = require('html-webpack-plugin');
  2. const CopyPlugin = require("copy-webpack-plugin");
  3. const HtmlMinimizerPlugin = require("html-minimizer-webpack-plugin");
  4. const { CleanWebpackPlugin } = require("clean-webpack-plugin");
  5. const MiniCssExtractPlugin = require("mini-css-extract-plugin");
  6. const CompressionPlugin = require("compression-webpack-plugin");
  7. const TerserPlugin = require("terser-webpack-plugin");
  8. const CssMinimizerPlugin = require("css-minimizer-webpack-plugin");
  9. const ImageMinimizerPlugin = require("image-minimizer-webpack-plugin");
  10. const webpack = require("webpack");
  11. const path = require("path");
  12. const BundleAnalyzerPlugin = require('webpack-bundle-analyzer').BundleAnalyzerPlugin;
  13. const globSync = require("glob").sync;
  14. const glob = require('glob');
  15. const { merge } = require('webpack-merge');
  16. const devserver = require('./webpack/webpack.dev.js');
  17. const fs = require('fs');
  18. const zlib = require("zlib");
  19. const PurgeCSSPlugin = require('purgecss-webpack-plugin')
  20. const PATHS = {
  21. src: path.join(__dirname, 'src')
  22. }
  23. class BuildEventsHook {
  24. constructor(name, fn, stage = 'afterEmit') {
  25. this.name = name;
  26. this.stage = stage;
  27. this.function = fn;
  28. }
  29. apply(compiler) {
  30. compiler.hooks[this.stage].tap(this.name, this.function);
  31. }
  32. }
  33. module.exports = (env, options) => (
  34. merge(env.WEBPACK_SERVE ? devserver : {},
  35. {
  36. entry:
  37. {
  38. index: './src/index.ts'
  39. },
  40. module: {
  41. rules: [
  42. {
  43. test: /\.ejs$/,
  44. loader: 'ejs-loader',
  45. options: {
  46. variable: 'data',
  47. interpolate : '\\{\\{(.+?)\\}\\}',
  48. evaluate : '\\[\\[(.+?)\\]\\]'
  49. }
  50. },
  51. {
  52. test: /\.(woff(2)?|ttf|eot)(\?v=\d+\.\d+\.\d+)?$/,
  53. use: [
  54. {
  55. loader: 'file-loader',
  56. options: {
  57. name: '[name].[ext]',
  58. outputPath: 'fonts/'
  59. }
  60. }
  61. ]
  62. },
  63. // {
  64. // test: /\.s[ac]ss$/i,
  65. // use: [{
  66. // loader: 'style-loader', // inject CSS to page
  67. // },
  68. // {
  69. // loader: MiniCssExtractPlugin.loader,
  70. // options: {
  71. // publicPath: "../",
  72. // },
  73. // },
  74. // "css-loader",
  75. // {
  76. // loader: "postcss-loader",
  77. // options: {
  78. // postcssOptions: {
  79. // plugins: [["autoprefixer"]],
  80. // },
  81. // },
  82. // },
  83. // "sass-loader",
  84. // ]
  85. // },
  86. {
  87. test: /\.(scss)$/,
  88. use: [
  89. {
  90. loader: MiniCssExtractPlugin.loader,
  91. options: {
  92. publicPath: "../",
  93. },
  94. },
  95. // {
  96. // // inject CSS to page
  97. // loader: 'style-loader'
  98. // },
  99. {
  100. // translates CSS into CommonJS modules
  101. loader: 'css-loader'
  102. },
  103. {
  104. // Run postcss actions
  105. loader: 'postcss-loader',
  106. options: {
  107. // `postcssOptions` is needed for postcss 8.x;
  108. // if you use postcss 7.x skip the key
  109. postcssOptions: {
  110. // postcss plugins, can be exported to postcss.config.js
  111. plugins: function () {
  112. return [
  113. require('autoprefixer')
  114. ];
  115. }
  116. }
  117. }
  118. }, {
  119. // compiles Sass to CSS
  120. loader: 'sass-loader'
  121. }]
  122. },
  123. {
  124. test: /\.js$/,
  125. exclude: /(node_modules|bower_components)/,
  126. use: {
  127. loader: "babel-loader",
  128. options: {
  129. presets: ["@babel/preset-env"],
  130. },
  131. },
  132. },
  133. {
  134. test: /\.(jpe?g|png|gif|svg)$/i,
  135. type: "asset",
  136. },
  137. // {
  138. // test: /\.html$/i,
  139. // type: "asset/resource",
  140. // },
  141. {
  142. test: /\.html$/i,
  143. loader: "html-loader",
  144. options: {
  145. minimize: true,
  146. }
  147. },
  148. {
  149. test: /\.tsx?$/,
  150. use: 'ts-loader',
  151. exclude: /node_modules/,
  152. },
  153. ],
  154. },
  155. plugins: [
  156. new HtmlWebpackPlugin({
  157. title: 'SqueezeESP32',
  158. template: './src/index.ejs',
  159. filename: 'index.html',
  160. inject: 'body',
  161. minify: {
  162. html5 : true,
  163. collapseWhitespace : true,
  164. minifyCSS : true,
  165. minifyJS : true,
  166. minifyURLs : false,
  167. removeAttributeQuotes : true,
  168. removeComments : true, // false for Vue SSR to find app placeholder
  169. removeEmptyAttributes : true,
  170. removeOptionalTags : true,
  171. removeRedundantAttributes : true,
  172. removeScriptTypeAttributes : true,
  173. removeStyleLinkTypeAttributese : true,
  174. useShortDoctype : true
  175. },
  176. favicon: "./src/assets/images/favicon-32x32.png",
  177. excludeChunks: ['test'],
  178. }),
  179. // new CompressionPlugin({
  180. // test: /\.(js|css|html|svg)$/,
  181. // //filename: '[path].br[query]',
  182. // filename: "[path][base].br",
  183. // algorithm: 'brotliCompress',
  184. // compressionOptions: { level: 11 },
  185. // threshold: 100,
  186. // minRatio: 0.8,
  187. // deleteOriginalAssets: false
  188. // }),
  189. new MiniCssExtractPlugin({
  190. filename: "css/[name].[contenthash].css",
  191. }),
  192. new PurgeCSSPlugin({
  193. paths: glob.sync(`${PATHS.src}/**/*`, { nodir: true }),
  194. }),
  195. new webpack.ProvidePlugin({
  196. $: "jquery",
  197. jQuery: "jquery",
  198. "window.jQuery": "jquery",
  199. Popper: ["popper.js", "default"],
  200. Util: "exports-loader?Util!bootstrap/js/dist/util",
  201. Dropdown: "exports-loader?Dropdown!bootstrap/js/dist/dropdown",
  202. }),
  203. new CompressionPlugin({
  204. //filename: '[path].gz[query]',
  205. test: /\.js$|\.css$|\.html$/,
  206. filename: "[path][base].gz",
  207. algorithm: 'gzip',
  208. threshold: 100,
  209. minRatio: 0.8,
  210. }),
  211. new BuildEventsHook('Update C App',
  212. function (stats, arguments) {
  213. if (options.mode !== "production") return;
  214. let wifiManagerPath=path.posix.resolve(process.cwd(),'..','..');
  215. let buildCRootPath=path.posix.resolve(process.cwd(),'..','..','..','..');
  216. let distPath = path.posix.resolve('dist');
  217. let posixWebAppPath=path.posix.resolve(process.cwd(),'..');
  218. fs.appendFileSync('./dist/index.html.gz',
  219. zlib.gzipSync(fs.readFileSync('./dist/index.html'),
  220. {
  221. chunckSize: 65536,
  222. level: zlib.constants.Z_BEST_COMPRESSION
  223. }));
  224. var getDirectories = function (src, callback) {
  225. var searchPath = path.posix.join(src, '/**/*(*.gz|favicon-32x32.png)');
  226. console.log(`Post build: Getting file list from ${searchPath}`);
  227. glob(searchPath, callback);
  228. };
  229. var cleanUpPath = path.posix.join(buildCRootPath, '/build/*.S');
  230. console.log(`Post build: Cleaning up previous builds in ${cleanUpPath}`);
  231. glob(cleanUpPath, function (err, list) {
  232. if (err) {
  233. console.error('Error', err);
  234. } else {
  235. list.forEach(fileName => {
  236. try {
  237. console.log(`Post build: Purging old binary file ${fileName} from C project.`);
  238. fs.unlinkSync(fileName)
  239. //file removed
  240. } catch (ferr) {
  241. console.error(ferr)
  242. }
  243. });
  244. }
  245. },
  246. 'afterEmit'
  247. );
  248. console.log('Generating C include files from webpack build output');
  249. getDirectories('./dist', function (err, list) {
  250. console.log(`Post build: found ${list.length} files. Relative path: ${wifiManagerPath}.`);
  251. if (err) {
  252. console.log('Error', err);
  253. } else {
  254. let exportDefHead =
  255. `/***********************************
  256. webpack_headers
  257. ${arguments[1]}
  258. ***********************************/
  259. #pragma once
  260. #include <inttypes.h>
  261. extern const char * resource_lookups[];
  262. extern const uint8_t * resource_map_start[];
  263. extern const uint8_t * resource_map_end[];`;
  264. let exportDef = '// Automatically generated. Do not edit manually!.\n' +
  265. '#include <inttypes.h>\n';
  266. let lookupDef = 'const char * resource_lookups[] = {\n';
  267. let lookupMapStart = 'const uint8_t * resource_map_start[] = {\n';
  268. let lookupMapEnd = 'const uint8_t * resource_map_end[] = {\n';
  269. let cMake='';
  270. list.forEach(foundFile => {
  271. let exportName = path.basename(foundFile).replace(/[\. \-]/gm, '_');
  272. let cmakeFileName = path.posix.relative(wifiManagerPath,foundFile);
  273. let httpRelativePath=path.posix.join('/',path.posix.relative('dist',foundFile));
  274. exportDef += `extern const uint8_t _${exportName}_start[] asm("_binary_${exportName}_start");\nextern const uint8_t _${exportName}_end[] asm("_binary_${exportName}_end");\n`;
  275. lookupDef += `\t"${httpRelativePath}",\n`;
  276. lookupMapStart += '\t_' + exportName + '_start,\n';
  277. lookupMapEnd += '\t_' + exportName + '_end,\n';
  278. cMake += `target_add_binary_data( __idf_wifi-manager ${cmakeFileName} BINARY)\n`;
  279. console.log(`Post build: adding cmake file reference to ${cmakeFileName} from C project, with web path ${httpRelativePath}.`);
  280. });
  281. lookupDef += '""\n};\n';
  282. lookupMapStart = lookupMapStart.substring(0, lookupMapStart.length - 2) + '\n};\n';
  283. lookupMapEnd = lookupMapEnd.substring(0, lookupMapEnd.length - 2) + '\n};\n';
  284. try {
  285. fs.writeFileSync('webapp.cmake', cMake);
  286. fs.writeFileSync('webpack.c', exportDef + lookupDef + lookupMapStart + lookupMapEnd);
  287. fs.writeFileSync('webpack.h', exportDefHead);
  288. //file written successfully
  289. } catch (e) {
  290. console.error(e);
  291. }
  292. }
  293. });
  294. console.log('Post build completed.');
  295. })
  296. ],
  297. optimization: {
  298. minimize: true,
  299. minimizer: [
  300. new TerserPlugin(),
  301. new HtmlMinimizerPlugin(),
  302. new CssMinimizerPlugin(),
  303. new ImageMinimizerPlugin({
  304. minimizer: {
  305. implementation: ImageMinimizerPlugin.imageminMinify,
  306. options: {
  307. // Lossless optimization with custom option
  308. // Feel free to experiment with options for better result for you
  309. plugins: [
  310. ["gifsicle", { interlaced: true }],
  311. ["jpegtran", { progressive: true }],
  312. ["optipng", { optimizationLevel: 5 }],
  313. // Svgo configuration here https://github.com/svg/svgo#configuration
  314. [
  315. "svgo",
  316. {
  317. plugins: [
  318. {
  319. name: 'preset-default',
  320. params: {
  321. overrides: {
  322. // customize default plugin options
  323. inlineStyles: {
  324. onlyMatchedOnce: false,
  325. },
  326. // or disable plugins
  327. removeDoctype: false,
  328. },
  329. },
  330. }
  331. ],
  332. },
  333. ],
  334. ],
  335. },
  336. },
  337. }),
  338. //new BundleAnalyzerPlugin()
  339. ],
  340. // splitChunks: {
  341. // cacheGroups: {
  342. // styles: {
  343. // name: 'styles',
  344. // test: /\.css$/,
  345. // chunks: 'all',
  346. // enforce: true
  347. // }
  348. // }
  349. // }
  350. },
  351. // output: {
  352. // filename: "[name].js",
  353. // path: path.resolve(__dirname, "dist"),
  354. // publicPath: "",
  355. // },
  356. resolve: {
  357. extensions: ['.tsx', '.ts', '.js', '.ejs' ],
  358. },
  359. output: {
  360. path: path.resolve(__dirname, 'dist'),
  361. filename: './js/[name].[fullhash:6].bundle.js',
  362. clean: true
  363. },
  364. }
  365. )
  366. );