webpack 5 和 shakapacker 7:jQuery 未初始化

webpack 5 & shakapacker 7: jQuery not being initialized

提问人:s0rin 提问时间:9/18/2023 最后编辑:s0rin 更新时间:9/19/2023 访问量:144

问:

将 webpack 4.46.0 升级到 webpack 5.88.2 和 shakapacker 7.0.3 (rails 6.1.7.6) 后出现错误。以下是当前的设置:Uncaught ReferenceError: $ is not defined

package.json

{
  "name": "app",
  "private": true,
  "dependencies": {
    "@babel/runtime": "^7.22.15",
    "@fortawesome/fontawesome-free": "^6.4.2",
    "@hotwired/turbo-rails": "^7.3.0",
    "@popperjs/core": "^2.11.8",
    "@rails/ujs": "^7.0.8",
    "babel-loader": "^9.1.3",
    "bootstrap": "^5.3.1",
    "core-js": "^3.32.2",
    "corejs-typeahead": "^1.3.3",
    "css-loader": "^6.8.1",
    "exports-loader": "^4.0.0",
    "expose-loader": "^4.1.0",
    "file-loader": "^6.2.0",
    "flag-icons": "^6.11.0",
    "handlebars": "^4.7.8",
    "inflection": "^2.0.1",
    "jbuilder": "^0.0.5",
    "jquery": "^3.7.1",
    "jquery-ui": "^1.13.2",
    "jquery-ui-dist": "^1.13.2",
    "jquery-ujs": "^1.2.3",
    "jstree": "^3.3.15",
    "mini-css-extract-plugin": "^2.7.6",
    "moment": "^2.29.4",
    "popper.js": "^1.16.1",
    "postcss": "^8.4.29",
    "rails-erb-loader": "^5.5.2",
    "sass-loader": "^13.3.2",
    "shakapacker": "^7.0.3",
    "sidekiq": "^1.1.1",
    "style-loader": "^3.3.3",
    "webpack": "^5.88.2",
    "webpack-assets-manifest": "^5.1.0",
    "webpack-cli": "^5.1.4",
    "webpack-dev-server": "^4.15.1",
    "webpack-merge": "^5.9.0",
    "webpack-sources": "^3.2.3",
    "yarn": "^1.22.19"
  },
  "babel": {
    "presets": [
      "./node_modules/shakapacker/package/babel/preset.js"
    ]
  },
  "devDependencies": {
    "@babel/core": "^7.22.19",
    "@babel/plugin-proposal-class-properties": "^7.18.6",
    "@babel/plugin-proposal-object-rest-spread": "^7.20.7",
    "@babel/plugin-proposal-private-methods": "^7.18.6",
    "@babel/plugin-proposal-private-property-in-object": "^7.21.11",
    "@babel/plugin-syntax-dynamic-import": "^7.8.3",
    "@babel/plugin-transform-destructuring": "^7.22.15",
    "@babel/plugin-transform-regenerator": "^7.22.10",
    "@babel/plugin-transform-runtime": "^7.22.15",
    "@babel/preset-env": "^7.22.15",
    "babel-plugin-macros": "^3.1.0",
    "clean-webpack-plugin": "^4.0.0",
    "compression-webpack-plugin": "^10.0.0",
    "moment-timezone": "^0.5.43",
    "regenerator-runtime": "^0.14.0",
    "sass": "^1.67.0",
    "stylus": "^0.60.0",
    "terser-webpack-plugin": "^5.3.9",
    "url-loader": "^4.1.1"
  }
}

config/webpack/webpack.config.js

const { generateWebpackConfig } = require('shakapacker')
const webpackConfig = generateWebpackConfig()
const customConfig = require('./custom')
module.exports = generateWebpackConfig()

config/webpack/custom.js

 var path = require('path');
 var webpack = require('webpack');
 const MiniCssExtractPlugin = require("mini-css-extract-plugin");
 const devMode = (process.env.NODE_ENV !== "staging") && (process.env.NODE_ENV !== "production");

 module.exports = {
   mode: 'development',
   entry: '../../app/javascript/packs/application.js',
   output: {
     path: path.resolve(__dirname, '../public'),
     publicPath: 'public/',
     filename: '[name].js',   // could be omitted, that's the default
     assetModuleFilename: 'public/images/[name].[ext]'
   },
   resolve: {
     modules: ['node_modules'],
     extensions: ['.css', '.sass', '.scss', '.js', '.json'],
     alias: {
       jquery: 'jquery',
       'jquery-ui': 'jquery-ui/jquery-ui.js',
       typeahead: 'core-typeahead',
     }
   },
   module: {
     rules: [
       {
         test: /\.html\.erb$/,
         loader: 'rails-erb-loader'
       },
       {
         test: require.resolve('jquery'),
         loader: 'expose-loader',
         options: {
           exposes: ['$', 'jQuery'],
         },
       },
       {
         test: /\.s[ac]ss$/i,
         use: [
           devMode ? "style-loader" : MiniCssExtractPlugin.loader,
           "css-loader",
           "postcss-loader",
           "sass-loader",
           "style-loader",
         ],
       },
       {
         test: /\.(jpe?g|png|gif|svg|gif|png|jpg|eot|ttf|otf|woff|woff2)$/i,
         loader: 'file-loader'
       },
     ]
   },
   performance: {
     hints: false,
     maxEntrypointSize: 512000,
     maxAssetSize: 512000
   },
   optimization: {
     splitChunks: {
       chunks: 'all'
     }
   },
   progress: true,
   stats: {
     errorDetails: true, //this does show errors
     colors: true,
     modules: true,
     reasons: true
   }
 }

config/webpack/environment.js

const webpack = require('webpack');
const { generateWebpackConfig } = require('shakapacker')
const erb = require('./loaders/erb');

const customConfig = {
  resolve: {
    fallback: {
      dgram: false,
      fs: false,
      net: false,
      tls: false,
      child_process: false
    }
  }
};

environment.plugins.prepend('Provide', 
  new webpack.ProvidePlugin({
    $: 'jquery',
    jQuery: 'jquery',
    jquery: 'jquery',
    Popper: ['popper.js', 'default'],
    Rails: ['@rails/ujs'],
    moment: 'moment'
  })
);

environment.loaders.prepend('erb', erb);
environment.config.merge(customConfig);
module.exports = environment;

config/webpack/development.js

process.env.NODE_ENV = process.env.NODE_ENV || 'development'
const environment = require('./environment')
module.exports = environment.toWebpackConfig()

应用程序/资产/javascript/包/应用程序.js

...
import $ from 'jquery';
window.jQuery = $;
window.$ = $;
import 'jquery-ui-dist/jquery-ui';
...

应用程序/视图/布局/应用程序.html.erb

...
<%= stylesheet_pack_tag 'application', :media => "all", :data => {:turbo => {:track => 'reload'}} %>
<%= javascript_pack_tag 'application', :data => {:turbo => {:track => 'reload'}, :defer => false} %>
...

没有发生编译错误,任何线索为什么jquery不加载?

jQuery Ruby-on-Rails Webpack-5 Shakapacker

评论


答:

1赞 Mostafa Ahangarha 9/19/2023 #1

请查看 Shakapacker README 文件v6 升级文档,其中提出了两种方法。我在这里引用它们:

  1. expose-loader

    如果使用 全局公开 jquery,则在 中使用 ,将选项传递给 .expose-loaderimport $ from "expose-loader?exposes=$,jQuery!jquery"app/javascript/application.jsdefer: falsejavascript_pack_tag

  2. 在 webpack 配置中使用:resolve.alias

    // config/webpack/custom.js
    module.exports = {
      resolve: {
        alias: {
          jquery: 'jquery/src/jquery',
        }
      }
    }
    
  3. 在 webpack 配置中使用 ProvidePlugin

    
    module.exports = {
      // ...
      plugins: [
        new webpack.ProvidePlugin({
          $: 'jquery',
          jQuery: 'jquery'
        })
      ]
      // ...
    };
    
1赞 s0rin 9/19/2023 #2

我发现该错误与 webpacker/shakapacker 配置无关,而是与由于缺少“defer”属性而未正确加载的文件有关。js.erb

应用程序/视图/布局/应用程序.html.erb

...
<%= stylesheet_pack_tag 'application', :media => "all", :data => {:turbo => {:track => 'reload'}} %>
<%= javascript_pack_tag 'application', :data => {:turbo => {:track => 'reload'}} %>
<%= javascript_include_tag '/js/browse_jstree.js' %>
...

Webpacker 助手将生成设置为默认值的 HTML 代码(这是 webpacker 5 中的新功能)。javascript_pack_tagdefertrue

...
<script src="/packs/js/runtime-f910d2d79960fb33a2bb.js" data-turbo="{&quot;track&quot;:&quot;reload&quot;}" defer="defer"></script>
<script src="/packs/js/vendors-node_modules_hotwired_turbo_dist_turbo_es2017-esm_js-node_modules_rails_ujs_lib_asset-a849ed-49174ef89c67613e1ea1.js" data-turbo="{&quot;track&quot;:&quot;reload&quot;}" defer="defer"></script>
<script src="/packs/js/application-3f6a4df6f831f107218d.js" data-turbo="{&quot;track&quot;:&quot;reload&quot;}" defer="defer"></script>
<script src="/js/browse_jstree.js"></script>
...

因此,该文件不会等待脚本,它将在 jquery 之前加载。添加到已修复的问题js.erbdefer = truejavascript_include_tag

...
<%= javascript_include_tag '/js/browse_jstree.js', :defer => true %>
....

评论

0赞 Mostafa Ahangarha 9/20/2023
感谢您分享问题的根本原因。我非常感谢。我回答的第一项是关于同一件事。很高兴看到您的问题得到解决。