提问人:s0rin 提问时间:9/18/2023 最后编辑:s0rin 更新时间:9/19/2023 访问量:144
webpack 5 和 shakapacker 7:jQuery 未初始化
webpack 5 & shakapacker 7: jQuery not being initialized
问:
将 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不加载?
答:
请查看 Shakapacker README 文件和 v6 升级文档,其中提出了两种方法。我在这里引用它们:
用
expose-loader
如果使用 全局公开 jquery,则在 中使用 ,将选项传递给 .
expose-loader
import $ from "expose-loader?exposes=$,jQuery!jquery"
app/javascript/application.js
defer: false
javascript_pack_tag
在 webpack 配置中使用:
resolve.alias
// config/webpack/custom.js module.exports = { resolve: { alias: { jquery: 'jquery/src/jquery', } } }
在 webpack 配置中使用 ProvidePlugin
module.exports = { // ... plugins: [ new webpack.ProvidePlugin({ $: 'jquery', jQuery: 'jquery' }) ] // ... };
我发现该错误与 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_tag
defer
true
...
<script src="/packs/js/runtime-f910d2d79960fb33a2bb.js" data-turbo="{"track":"reload"}" 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="{"track":"reload"}" defer="defer"></script>
<script src="/packs/js/application-3f6a4df6f831f107218d.js" data-turbo="{"track":"reload"}" defer="defer"></script>
<script src="/js/browse_jstree.js"></script>
...
因此,该文件不会等待脚本,它将在 jquery 之前加载。添加到已修复的问题js.erb
defer = true
javascript_include_tag
...
<%= javascript_include_tag '/js/browse_jstree.js', :defer => true %>
....
评论