Flag of Ukraine
SymfonyCasts stands united with the people of Ukraine

jQuery & Legacy JS in your Template

Keep on Learning!

If you liked what you've learned so far, dive in!
Subscribe to get access to this tutorial plus
video, code and script downloads.

Start your All-Access Pass
Buy just this tutorial for $12.00

With a Subscription, click any sentence in the script to jump to that part of the video!

Login Subscribe

Here's a simple challenge: when I hover over the leaderboard, I want to show a JavaScript Bootstrap popover that gives us some info about how it works.

Open up the template: app/Resources/views/lift/index.html.twig. Add a class on the h2 that we can target in JavaScript: js-custom-popover. Then, I'll add a few other things that will help the popover: data-toggle="popover", a title and data-content:

... lines 1 - 2
{% block body %}
<div class="row">
... lines 5 - 34
<div class="col-md-5">
<div class="leaderboard">
<h2 class="text-center js-custom-popover" data-toggle="popover" title="About the Leaderboard" data-content="Want to be the leader? Lift stuff!">
... lines 38 - 39
</h2>
... lines 41 - 42
</div>
</div>
</div>
{% endblock %}
... lines 47 - 62

To get this working, all we need to do is write some JavaScript to find that element and activate the popover. Normally, we would put this in rep_log.js. In fact, that's the best place to put it. Well, there, or some file that it requires. The point is, now that our JavaScript has matured, 100% of your JS code should live in an external JavaScript file. You should have no JavaScript in your templates, other than the one script tag:

... lines 1 - 47
{% block javascripts %}
... lines 49 - 50
<script src="{{ asset('build/rep_log.js') }}"></script>
{% endblock %}

But, what if you're using Webpack on a legacy app... that still has a ton of JavaScript in your templates? You should refactor this. But, since that can be a big job, let's find out how we can still write JavaScript in a template... even though we shouldn't.

Add a new script tag right in the template. Here, code like normally: add a $(document).ready() block, then find our .js-custom-popover elements and activate popover(). I'll pass that a few options:

... lines 1 - 47
{% block javascripts %}
... lines 49 - 50
<script src="{{ asset('build/rep_log.js') }}"></script>
<script>
$(document).ready(function() {
$('.js-custom-popover').popover({
trigger: 'hover',
placement: 'left'
});
});
</script>
{% endblock %}

So simple! It should work, right? After all, rep_log.js requires jQuery, so we should be able to use it in our template.

Well, try it. Refresh the page. Uh oh... error!

$ is not defined

This is coming from our template. What's going on? We are requiring jQuery!

const $ = require('jquery');
... lines 2 - 8

Why jQuery / $ is not Global Anymore

Well... we already know what's going on. When we require jquery, it detects that it is being used in a module environment. So it does not set a global jQuery or $ variable. Instead, it just returns the jQuery function, which we set to $. But that only affects this file.

And this is good! We want each module to live in isolation like this. But, if you do need to expose a jQuery variable globally, you can.

First, since we're using a Bootstrap function on this page, require('bootstrap-sass') so the popover function is available:

const $ = require('jquery');
require('bootstrap-sass');
... lines 3 - 13

Making jQuery / $ Global

Now, to actually make $ available as a global variable, we can set it on the window object: window.$ = $:

const $ = require('jquery');
require('bootstrap-sass');
... lines 3 - 4
// expose $ globally so I can use it in the template
// ... even though I should put all my code here!
window.$ = $;
... lines 8 - 13

That's it! But remember, this is not ideal, and is just meant to support legacy code.

Refresh now! Ha! It works!

The ProvidePlugin Complication

But, I have one tiny complication. In webpack.config.js, near the bottom, we added something called the webpack.ProvidePlugin():

107 lines webpack.config.js
... lines 1 - 30
module.exports = {
... lines 32 - 93
plugins: [
new webpack.ProvidePlugin({
jQuery: 'jquery',
$: 'jquery'
}),
... lines 99 - 103
],
... line 105
};

Thanks to this, if any JavaScript relies on jQuery or $ to be available as a global variable, Webpack rewrites that code to require the jquery module properly. This does not make the jQuery or $ variables available globally, like in a template. Instead, it fixes code to not need that. If you do want to make these variables global, that's what we just did in rep_log.js:

... lines 1 - 4
// expose $ globally so I can use it in the template
// ... even though I should put all my code here!
window.$ = $;
... lines 8 - 13

In other words, these are solving two separate, but similar issues.

Here's the complication: sometimes, a third party library will try to reference jQuery globally not by just saying jQuery, but by saying window.jQuery... which is really the same thing. But, right now, the ProvidePlugin would not fix that situation. So we could still have some issues with some libraries.

No worries! Add two more lines: 'window.jQuery': 'jquery' and 'window.$': 'jquery':

109 lines webpack.config.js
... lines 1 - 30
module.exports = {
... lines 32 - 93
plugins: [
new webpack.ProvidePlugin({
jQuery: 'jquery',
$: 'jquery',
'window.jQuery': 'jquery',
'window.$': 'jquery',
}),
... lines 101 - 107
};

Cool! So now we're covered. Re-start Webpack:

./node_modules/.bin/webpack --watch

And, refresh! Woh! What! An error! Once again, it says that $ is not defined! What the heck!

Using global to set Global Variables

This is a total gotcha. Thanks to the ProvidePlugin(), it's actually now re-writing our code, changing the window.$ in a way so that it does not actually create a global variable anymore. This is not what we intended!

The fix is easy: change window.$ to global.$:

... lines 1 - 4
// expose $ globally so I can use it in the template
// ... even though I should put all my code here!
global.$ = $;
... lines 8 - 13

What? We know that window is the global variable in a browser environment. Well, in Webpack, you're allowed to set things on a global variable. When you do, it figures out that you're in a browser environment, and sets that on window. Basically, this does the same thing as before, but won't be re-written by the ProvidePlugin().

Just by making this change, when we refresh, it works. Now, in practice, some libraries do reference window.jQuery, but I don't know of any that reference window.$, so you can remove it:

108 lines webpack.config.js
... lines 1 - 30
module.exports = {
... lines 32 - 93
plugins: [
new webpack.ProvidePlugin({
jQuery: 'jquery',
$: 'jquery',
'window.jQuery': 'jquery',
}),
... lines 100 - 106
};

Next! Let's talk about import and export... the more hipster alternativse to require and module.exports.

Leave a comment!

2
Login or Register to join the conversation
Default user avatar
Default user avatar Дмитрий Политов | posted 5 years ago

Thanks for this excellent tutorial !!!

Reply
Cat in space

"Houston: no signs of life"
Start the conversation!

This tutorial explains the concepts of an old version of Webpack using an old version of Symfony. The most important concepts are still the same, but you should expect significant differences in new versions.

What PHP libraries does this tutorial use?

// composer.json
{
    "require": {
        "php": "^7.2.0",
        "symfony/symfony": "3.3.*", // v3.3.16
        "twig/twig": "2.10.*", // v2.10.0
        "doctrine/orm": "^2.5", // v2.7.0
        "doctrine/doctrine-bundle": "^1.6", // 1.10.3
        "doctrine/doctrine-cache-bundle": "^1.2", // 1.3.5
        "symfony/swiftmailer-bundle": "^2.3", // v2.6.3
        "symfony/monolog-bundle": "^2.8", // v2.12.1
        "symfony/polyfill-apcu": "^1.0", // v1.4.0
        "sensio/distribution-bundle": "^5.0", // v5.0.22
        "sensio/framework-extra-bundle": "^3.0.2", // v3.0.26
        "incenteev/composer-parameter-handler": "^2.0", // v2.1.2
        "friendsofsymfony/user-bundle": "^2.0", // v2.1.2
        "doctrine/doctrine-fixtures-bundle": "~2.3", // v2.4.1
        "doctrine/doctrine-migrations-bundle": "^1.2", // v1.3.2
        "composer/package-versions-deprecated": "^1.11", // 1.11.99
        "friendsofsymfony/jsrouting-bundle": "^1.6" // 1.6.0
    },
    "require-dev": {
        "sensio/generator-bundle": "^3.0", // v3.1.6
        "symfony/phpunit-bridge": "^3.0" // v3.3.5
    }
}

What JavaScript libraries does this tutorial use?

// package.json
{
    "dependencies": [],
    "devDependencies": {
        "babel-core": "^6.25.0", // 6.25.0
        "babel-loader": "^7.1.1", // 7.1.1
        "babel-plugin-syntax-dynamic-import": "^6.18.0", // 6.18.0
        "babel-preset-env": "^1.6.0", // 1.6.0
        "bootstrap-sass": "^3.3.7", // 3.3.7
        "clean-webpack-plugin": "^0.1.16", // 0.1.16
        "copy-webpack-plugin": "^4.0.1", // 4.0.1
        "core-js": "^2.4.1", // 2.4.1
        "css-loader": "^0.28.4", // 0.28.4
        "extract-text-webpack-plugin": "^3.0.0", // 3.0.0
        "file-loader": "^0.11.2", // 0.11.2
        "font-awesome": "^4.7.0", // 4.7.0
        "jquery": "^3.2.1", // 3.2.1
        "lodash": "^4.17.4", // 4.17.4
        "node-sass": "^4.5.3", // 4.5.3
        "resolve-url-loader": "^2.1.0", // 2.1.0
        "sass-loader": "^6.0.6", // 6.0.6
        "style-loader": "^0.18.2", // 0.18.2
        "sweetalert2": "^6.6.6", // 6.6.6
        "webpack": "^3.4.1", // 3.4.1
        "webpack-chunk-hash": "^0.4.0", // 0.4.0
        "webpack-dev-server": "^2.6.1", // 2.6.1
        "webpack-manifest-plugin": "^1.2.1" // 1.2.1
    }
}
userVoice