phoebe
jQuery in Phoenix 1.4 / Webpack
Hi,
Following the Programming Phoenix 1.4 beta book, I’d like to use a simple jQuery selector in my form.html.eex template. I have a few questions as to what is the best approach for this:
-
In which folder is it best to save the js file. If I have a Context/Model, e.g.
Multimedia/Video- and aform.html.eextemplate forVideo, where should I save the js code needed for this page? I prefer to save the js code near to the page on which it will be used, so I would have it in a folder such astemplates/video/jsbut it seems that I have to put them inassets/jsfor webpack to know where to look. Is it common practice out there to do this? -
In anycase, my templates cannot find the jquery
$even though in myapp.jsI have this imported. I have put a simple<script>tag in theform.html.eextemplate which will use the$jQuery selector, and I get an error saying$ is not defined- why is this? I thought $ was global when copied topriv/static/js/app.js?
Thanks
Most Liked
kokolegorille
If You look at the application layout, the link to app.js is at the very end… which is usual for better loading performance. But the template are defined above, so they don’t know about $, because it will be defined later.
One solution I use is to put a placeholder for script just after loading app.js, like this…
<script type="text/javascript" src="<%= Routes.static_url(@conn, "/js/app.js") %>"></script>
<%= render_existing @view_module, "scripts.html", assigns %>
And in the corresponding view, I put
def render("scripts.html", _assigns) do
~E(<script src="/js/page_tree.js"></script>)
end
Scripts loaded this way have access to $.
But if You want to put javascript code inside template, You would probably need another solution.
It is not tested, but I think I would do it this way…
- Split bundle in vendor.js and app.js, putting jquery and related into vendor.
- Load vendor before rendering template.
By the way I use provide plugin in webpack config to load jquery.
new Webpack.ProvidePlugin({ // inject ES5 modules as global vars
$: 'jquery',
jQuery: 'jquery', 'window.jQuery': 'jquery',
Popper: ['popper.js', 'default']
})
PS. One quick test would be to move the loading of app.js before rendering template…
kokolegorille
I put this file under /assets/static/js/
This way it will copied automatically to priv… (by CopyWebpackPlugin)
new CopyWebpackPlugin([{ from: 'static/', to: './' }]),
I don’t need to declare page_tree.js in webpack, but if it is declared after app.js, it will benefit of all the stuff You loaded in app.js.
PS. You can also define multiple entries in webpack.config.js, and use [name].js as output… This would bundle any js through webpack.
OvermindDL1
For note, at least Chrome (and I think Firefox and Edge too) all define $ if not otherwise defined and it works like a selector query from jQuery, a convenient helper.
To see if $ is actually defined, test is on window.$ instead.







