-
-
Notifications
You must be signed in to change notification settings - Fork 198
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow to add custom loaders #11
Conversation
lib/WebpackConfig.js
Outdated
@@ -49,6 +49,7 @@ class WebpackConfig { | |||
this.providedVariables = {}; | |||
this.babelConfigurationCallback = function() {}; | |||
this.useReact = false; | |||
this.loaders = new Set(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This does not need to be a Set. It could just be an array.
You create a new object inside addLoader
, so the uniqueness check of the Set will never exclude anything anyway. You're just using it as an array here
What is the advantage of this API: .addLoader(/\.hbs$/, 'handlebars-loader') over: .addLoader({ test: /\.hbs$/, loader: 'handlebars-loader' }) Is the slightly shorter API worth the complexity of "okay, I need to pass more options, so I need to change the second parameter to an object and the original string becomes the loader key in that object"? It might be preferable to not introduce a magic difference over webpack's API here. PS: speaking as a passerby who's interested in Encore, not a project member. |
yeah, I think we should either use |
lib/config-generator.js
Outdated
@@ -235,6 +235,12 @@ class ConfigGenerator { | |||
}); | |||
} | |||
|
|||
if (this.webpackConfig.loaders.size > 0) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
no need for this check. .forEach
works fine on empty arrays too
Having to pass the full object means the name can then just as well change to Having to construct the object manually (like the second example) doesn't feel clean though. // ...
.addRule( // or addLoader
Encore.Rule()
.setTest( /\.hbs$/)
.addLoader('handlebars-loader')
.addOption('debug', !Encore.isProduction())
.addExclude('node_modules')
) The benefit here is that we can then easily add support for all options for a rule, it reads easier and it fits in with the style of the rest of the config. WDYT? |
First, woohoo! This is one of the first PR's I was hoping for - it's an obvious extension point that's needed :). I would definitely to stick close to webpack here. So, I prefer @fvsch's suggestion: .addLoader({ test: /\.hbs$/, loader: 'handlebars-loader' }) But @pierredup is totally correct that then... this is nothing more than a rule. But, as far as I can tell, Assuming that's true, right now, "loaders" is more common language, so I would still want it to be called |
Thanks for the info regarding rules and loaders @weaverryan. I've never used Webpack 1, and only recently started using Webpack so some things are still a bit confusing. I've made the changes as requested, but I still feel that it would be more difficult to use if there isn't a cleaner interface to define a loader config. By passing the raw loader config, it takes away the 'clean & powerful API' that Encore provides. It means you need to know exactly how the Webpack config should look like in order to construct the object. It also might make it more difficult for newcomers that don't know Webpack to get started with custom loaders. The handlebars-loader for example just states "The following query (or config) options are supported" with a list of the options, and no indication of how to use those config options. So you then need to go back to the Webpack documentation to try and understand how you should pass a config value. Instead you can have a cleaner way to add loaders with the Symfony docs telling you "to add a custom loader, just add your file extension and the the loader you want to use to this function", or "construct this object with this options to create a loader config" (Note: I'm talking about a custom rule or loader class here, not the initial way of passing an object that I described in the PR description, which wasn't actually an implemented feature but just a side-effect of how the loader was stored) |
This is a fair point - passing the entire configuration object doesn't have a "smoothness" to it. But, speaking about the handlebars example, it does lower the "translation" between reading their docs and seeing the raw The good news is, we could merge the PR as-is (i.e. with this simple/raw API)... and still add the |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Very minor comments!
test/WebpackConfig.js
Outdated
|
||
config.addLoader({ 'test': /\.custom$/, 'loader': 'custom-loader' }); | ||
|
||
expect(Array.from(config.loaders)).to.deep.equals([{ 'test': /\.custom$/, 'loader': 'custom-loader' }]); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We could remove the Array.from
now (because it is an Array), right?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, fixed
addLoader(loader) { | ||
this.loaders.push(loader); | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We need to expose this in index.js
as well (that file/module holds the true public API). And I think we should (in that file) create addLoader()
and also addRule()
(which would just call addLoader()
)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, don't know how I missed this :) Functions added
👍 |
Thank you @pierredup! I've added a docs issue for this new method: symfony/symfony-docs#8053 |
Currently Encore only supports specific loaders, and adding additional loaders is not very intuitive and adds additional (ugly) code to the config, E.G we are using handlebars with the handlebars-loader and have to manually add the loader to the config:
So I propose allowing to add custom loaders through a simple
addLoader(test, loader)
function.This changes the above code to the following:
It is also possible to set custom config for the loaders by passing an object instead of a string:
I have also added an object to define include and exclude directories. There are just so many options for a loader that I'm not sure which ones to support. Include and exclude seemed like the most obvious. Instead of having too many parameters for the function, I opted for an object with the options as keys, which makes it easier to add support for more options, but the drawback is that it is less clear which options are supported.