webpack.config.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  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 PurgeCSSPlugin = require('purgecss-webpack-plugin')
  18. const whitelister = require('purgecss-whitelister');
  19. const GrpcToolsNodeProtocPlugin = require('./webpack/GrpcToolsNodeProtocPlugin.js');
  20. const buildRootPath = path.join(process.cwd(), '..', '..', '..');
  21. const wifiManagerPath = glob.sync(path.join(buildRootPath, 'components/**/wifi-manager*'))[0];
  22. const ComponentsPath = glob.sync(path.join(buildRootPath, 'components/'))[0];
  23. const buildCRootPath = glob.sync(buildRootPath)[0];
  24. const SPIFFSPath = glob.sync(path.join(buildRootPath, 'SPIFFS'))[0];
  25. const PATHS = {
  26. src: path.join(__dirname, 'src')
  27. }
  28. class BuildEventsHook {
  29. constructor(name, fn, stage = 'afterEmit') {
  30. this.name = name;
  31. this.stage = stage;
  32. this.function = fn;
  33. }
  34. apply(compiler) {
  35. compiler.hooks[this.stage].tap(this.name, this.function);
  36. }
  37. }
  38. module.exports = (env, options) => (
  39. merge(
  40. env.WEBPACK_SERVE ? devserver : {},
  41. env.ANALYZE_SIZE ? {
  42. plugins: [new BundleAnalyzerPlugin(
  43. {
  44. analyzerMode: 'static',
  45. generateStatsFile: true,
  46. statsFilename: 'stats.json',
  47. }
  48. )]
  49. } : {},
  50. // { stats: 'verbose', },
  51. {
  52. entry:
  53. {
  54. index: './src/index.ts'
  55. },
  56. devtool: "source-map",
  57. module: {
  58. rules: [
  59. {
  60. test: /\.ejs$/,
  61. loader: 'ejs-loader',
  62. options: {
  63. variable: 'data',
  64. interpolate: '\\{\\{(.+?)\\}\\}',
  65. evaluate: '\\[\\[(.+?)\\]\\]'
  66. }
  67. },
  68. {
  69. test: /\.(woff(2)?|ttf|eot)(\?v=\d+\.\d+\.\d+)?$/,
  70. use: [
  71. {
  72. loader: 'file-loader',
  73. options: {
  74. name: '[name].[ext]',
  75. outputPath: 'fonts/'
  76. }
  77. }
  78. ]
  79. },
  80. // {
  81. // test: /\.s[ac]ss$/i,
  82. // use: [{
  83. // loader: 'style-loader', // inject CSS to page
  84. // },
  85. // {
  86. // loader: MiniCssExtractPlugin.loader,
  87. // options: {
  88. // publicPath: "../",
  89. // },
  90. // },
  91. // "css-loader",
  92. // {
  93. // loader: "postcss-loader",
  94. // options: {
  95. // postcssOptions: {
  96. // plugins: [["autoprefixer"]],
  97. // },
  98. // },
  99. // },
  100. // "sass-loader",
  101. // ]
  102. // },
  103. {
  104. test: /\.(scss)$/,
  105. use: [
  106. {
  107. loader: MiniCssExtractPlugin.loader,
  108. options: {
  109. publicPath: "../",
  110. },
  111. },
  112. // {
  113. // // inject CSS to page
  114. // loader: 'style-loader'
  115. // },
  116. {
  117. // translates CSS into CommonJS modules
  118. loader: 'css-loader'
  119. },
  120. {
  121. // Run postcss actions
  122. loader: 'postcss-loader',
  123. options: {
  124. // `postcssOptions` is needed for postcss 8.x;
  125. // if you use postcss 7.x skip the key
  126. postcssOptions: {
  127. // postcss plugins, can be exported to postcss.config.js
  128. plugins: function () {
  129. return [
  130. require('autoprefixer')
  131. ];
  132. }
  133. }
  134. }
  135. }, {
  136. // compiles Sass to CSS
  137. loader: 'sass-loader'
  138. }]
  139. },
  140. {
  141. test: /\.js$/,
  142. exclude: /(node_modules|bower_components)/,
  143. use: {
  144. loader: "babel-loader",
  145. options: {
  146. presets: ["@babel/preset-env"],
  147. plugins: ['@babel/plugin-transform-runtime']
  148. },
  149. },
  150. },
  151. {
  152. test: /\.(jpe?g|png|gif|svg)$/i,
  153. type: "asset",
  154. },
  155. // {
  156. // test: /\.html$/i,
  157. // type: "asset/resource",
  158. // },
  159. {
  160. test: /\.html$/i,
  161. loader: "html-loader",
  162. options: {
  163. minimize: true,
  164. }
  165. },
  166. {
  167. test: /\.tsx?$/,
  168. use: 'ts-loader',
  169. exclude: /node_modules/,
  170. }
  171. ],
  172. },
  173. plugins: [
  174. new GrpcToolsNodeProtocPlugin({
  175. protoPaths: [`${path.join(ComponentsPath, 'spotify/cspot/bell/external/nanopb/generator/proto')}`,
  176. `${path.join(buildCRootPath, 'protobuf/proto')}`],
  177. protoSources: [`${path.join(buildCRootPath, 'protobuf/proto/*.proto')}`,
  178. `${path.join(ComponentsPath, 'spotify/cspot/bell/external/nanopb/generator/proto/*.proto')}`],
  179. outputDir: './src/js/proto'
  180. }
  181. ),
  182. new HtmlWebpackPlugin({
  183. title: 'SqueezeESP32',
  184. template: './src/index.ejs',
  185. filename: 'index.html',
  186. inject: 'body',
  187. minify: {
  188. html5: true,
  189. collapseWhitespace: true,
  190. minifyCSS: true,
  191. minifyJS: true,
  192. minifyURLs: false,
  193. removeAttributeQuotes: true,
  194. removeComments: true, // false for Vue SSR to find app placeholder
  195. removeEmptyAttributes: true,
  196. removeOptionalTags: true,
  197. removeRedundantAttributes: true,
  198. removeScriptTypeAttributes: true,
  199. removeStyleLinkTypeAttributese: true,
  200. useShortDoctype: true
  201. },
  202. favicon: "./src/assets/images/favicon-32x32.png",
  203. excludeChunks: ['test'],
  204. }),
  205. // new CompressionPlugin({
  206. // test: /\.(js|css|html|svg)$/,
  207. // //filename: '[path].br[query]',
  208. // filename: "[path][base].br",
  209. // algorithm: 'brotliCompress',
  210. // compressionOptions: { level: 11 },
  211. // threshold: 100,
  212. // minRatio: 0.8,
  213. // deleteOriginalAssets: false
  214. // }),
  215. new MiniCssExtractPlugin({
  216. filename: "css/[name].css",
  217. }),
  218. new PurgeCSSPlugin({
  219. keyframes: false,
  220. paths: glob.sync(`${path.join(__dirname, 'src')}/**/*`, {
  221. nodir: true
  222. }),
  223. whitelist: whitelister('bootstrap/dist/css/bootstrap.css')
  224. }),
  225. new webpack.ProvidePlugin({
  226. $: "jquery",
  227. // jQuery: "jquery",
  228. // "window.jQuery": "jquery",
  229. // Popper: ["popper.js", "default"],
  230. // Util: "exports-loader?Util!bootstrap/js/dist/util",
  231. // Dropdown: "exports-loader?Dropdown!bootstrap/js/dist/dropdown",
  232. }),
  233. new CompressionPlugin({
  234. //filename: '[path].gz[query]',
  235. test: /\.js$|\.css$|\.html$/,
  236. filename: "[path][base].gz",
  237. algorithm: 'gzip',
  238. threshold: 100,
  239. minRatio: 0.8,
  240. }),
  241. ],
  242. optimization: {
  243. minimize: true,
  244. providedExports: true,
  245. usedExports: true,
  246. minimizer: [
  247. new TerserPlugin({
  248. terserOptions: {
  249. format: {
  250. comments: false,
  251. },
  252. },
  253. extractComments: false,
  254. // enable parallel running
  255. parallel: true,
  256. }),
  257. new HtmlMinimizerPlugin({
  258. minimizerOptions: {
  259. removeComments: true,
  260. removeOptionalTags: true,
  261. }
  262. }
  263. ),
  264. new CssMinimizerPlugin(),
  265. new ImageMinimizerPlugin({
  266. minimizer: {
  267. implementation: ImageMinimizerPlugin.imageminMinify,
  268. options: {
  269. // Lossless optimization with custom option
  270. // Feel free to experiment with options for better result for you
  271. plugins: [
  272. ["gifsicle", { interlaced: true }],
  273. ["jpegtran", { progressive: true }],
  274. ["optipng", { optimizationLevel: 5 }],
  275. // Svgo configuration here https://github.com/svg/svgo#configuration
  276. [
  277. "svgo",
  278. {
  279. plugins: [
  280. {
  281. name: 'preset-default',
  282. params: {
  283. overrides: {
  284. // customize default plugin options
  285. inlineStyles: {
  286. onlyMatchedOnce: false,
  287. },
  288. // or disable plugins
  289. removeDoctype: false,
  290. },
  291. },
  292. }
  293. ],
  294. },
  295. ],
  296. ],
  297. },
  298. },
  299. }),
  300. ],
  301. splitChunks: {
  302. cacheGroups: {
  303. vendor: {
  304. name: "node_vendors",
  305. test: /[\\/]node_modules[\\/]/,
  306. chunks: "all",
  307. }
  308. }
  309. }
  310. },
  311. // output: {
  312. // filename: "[name].js",
  313. // path: path.resolve(__dirname, "dist"),
  314. // publicPath: "",
  315. // },
  316. resolve: {
  317. extensions: ['.tsx', '.ts', '.js', '.ejs'],
  318. },
  319. output: {
  320. path: path.resolve(__dirname, 'dist'),
  321. // filename: './js/[name].[hash:6].bundle.js',
  322. filename: './js/[name].bundle.js',
  323. clean: true
  324. },
  325. }
  326. )
  327. );