-
Notifications
You must be signed in to change notification settings - Fork 20.6k
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
ESM distribution / exports
in package.json
#4592
Comments
Thanks for the report. FWIW, you wouldn't ever need: import jQuery from './node_modules/jquery/dist/jquery-esm.js'; In fact, it could be quite fragile as npm dependencies may be hoisted. This is enough if we generate the import jQuery from 'jquery/dist/jquery-esm.js'; |
While yes, there is that fragility, it can still be useful. For example, I like to use it in repo demos and tests where it is just meant for development and can safely be used as a direct dependency. It should also be safe for applications which are known to be the root. With such paths, there is no need for bundling as one needs when pointing to |
Ah, right, I was thinking about a bundler use case and you mean native browser import, got it. If there's no need for bundling then we could just point to |
Regarding bundling, Rollup allows "esm" as a |
I don't find any file named jquery/dist/jquery-esm.js in the project, how to find / build it please? |
@florealcab : It doesn't exist at this point. That's what my proposal is for. Currently, you have to build it yourself. |
@brettz9 ok, is it possible for you to say me how can I do that? |
You need to use a bundler like Rollup (or parcel) or Webpack. The configuration depends on how jQuery has set things up internally. Though I haven't done it with jQuery myself, with jQuery not having external dependencies, Rollup will probably work for you, and it is rather easy to use: https://rollupjs.org/ though you'll need @rollup/plugin-node-resolve at least. |
jQuery already has a build process based on Rollup defined in https://github.com/jquery/jquery/blob/master/build/tasks/build.js. For now it only generates a regular script file & its minified version; this could be enhanced to generate an ESM version as well. One thing to consider is what files we'd like to generate & where to put them. We already generate a regular & minified file for the full build & for the Slim build - those are 4 files. All of them are then uploaded to our own CDN but also to 3rd party CDNs like the Google or Microsoft ones. Would the ESM file be uploaded to them all? Do we also need an ESM slim file? We need to be careful here. |
@mgol generally the ESM version would only be used by bundlers right? It might be a good start to simply only have the If people end up making use of this those might be good steps to take in the future? If you are on board with this approach, I would be happy to give ESM a shot. |
I don't use nodejs in my project, but I use ESM version for my dependencies. So a packaged version of Jquery should be useful for me (available in the zip file along with script version of jquery) :) |
@florealcab to confirm you want an ESM bundled version in the zip? or just the current CJS works for you? |
Not only. Node.js has experimental ESM support, docs at https://nodejs.org/api/esm.html; it'd be good to work good with that. I expect bundlers to also start supporting the We're not in a hurry here: jQuery 4.0 won't be out for at least a few more months so we have this time to think about the best approach. |
@mgol for sure. I should have said that CDN's would not have much use for ESM versions and it would only be those people requiring via node or a bundler would care about an ESM bundle. |
@roopakv : While per @mgol 's comment, jquery may have to weigh the costs in doing so, ESM distributions are not only for Node and bundlers. Browser demos which which to avoid polluting HTML with dependencies, keeping their code simple without build steps and preserving modularity can use it. (While this is less of an issue for jquery, unless perhaps a Deno equivalent to |
I second the usefulness of ESM versions for the browser, I often like to develop small projects without using npm or a bundler, I just grab the bundled ESM version from a CDN like jsdelivr. |
I second @SeanBannister, it would be great to have jQuery ESM Browser like Vue do it. Thx ! |
module
in package.json
exports
in package.json
I've updated to suggest that now that native ESM has come to Node (and the LTS versions all support it), the choice of |
jQuery seems to be the only global I need in my (no-bundler-used-or-needed) demos these days... |
@brettz9 This past comment of mine in this thread is still relevant: #4592 (comment). In particular, its second section. If we want to generate an ESM file, we'd probably also need the Slim version so that there's no discrepancy between a regular script version and an ESM one. If you (or anyone else) could post a complete proposal on what to put in |
Regarding a slim file, I think having the slim version as ESM would be helpful too, yes. With modularity and importing, one is given the freedom not only to not need to manage or follow the dependency chain in adding script tags, but to choose only what one needs. To me, it is like pre-built binaries--a good project will offer a good range of choices. Regarding what is needed from
Hope that gives something to go by. |
prolly missing some risk like hoisting or something... to the bottom of the |
While adding such a line does allow a named import to work, it doesn't help with the main draw of using (named) modules--avoiding globals and reducing the chance for global conflict. The jQuery file would still set |
true! though my main goal is to continue to use jQuery w/ projects that are now entirely ESM / certainly the Gets tricky w/ other pkgs expecting those globals (eg: jquery-ui, jquery.cookie -- even just-out bootstrap v5 will add the jQuery short/nice methods IFF the global window var is found). |
It's apparently not a technical challenge to add the support for modules no less since there is already a Rollup process in place as mentioned in #4592 (comment) (I don't know if that removes But you can get it to work for your needs with ESM-only projects already by just doing the following: import './node_modules/jquery/dist/jquery.js';
// Then use `jQuery` or `$` below. If you are using the likes of ESLint which might complain about undeclared glboals, you still would have to do: /* globals jQuery */
import './node_modules/jquery/dist/jquery.js'; ...but this does at least avoid the need to define dependencies out-of-band within HTML, offering the advantages that:
|
isn't it possible to add a |
Adding the |
I'm not entire sure since I haven't used this approach yet but I think you can use conditional exports to provide ESM exports without it being a breaking change. This approach allows the support of the old way "main": "lib/index.js",
"exports": {
"import": "./index.esm.mjs",
"require": "./lib/index.js"
}, See this article What does it take to support Node.js ESM? |
I've submitted an experimental proposal at #5122. Read the PR description for more details. |
If there are any issues with CORS (ex: hard requirement of a web server to run your javascript, very non-jquery) a similar project at https://htmx.org recently discovered a "2 liner" method of allowing both classic May be of interest to the jQuery project. |
@gnat I don’t see anything ESM-related in that PR. In order to expose a library as an ECMAScript Module, you need to use the Exporting as CommonJS may make it possible to import from a file if you use a bundler that supports it, which is pretty common. But it does nothing to support actual native ESM. jQuery already exposes itself via AMD, CommonJS & browser globals. |
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. Fixes jquerygh-4592
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. Fixes jquerygh-4592
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. The `module` field is removed. It would only be used by legacy bundlers with no `exports` support. Rigth now, this seems to be mostly Parcel <3. However, those bundlers can just import `jquery/dist/jquery.mjs` directly as there are no restrictions for in-package deep imports if `exports` are not supported; and it's better to avoid definining legacy fields in new setups. Fixes jquerygh-4592
PR: #5255 |
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. The `module` field is removed. It would only be used by legacy bundlers with no `exports` support. Rigth now, this seems to be mostly Parcel <3. However, those bundlers can just import `jquery/dist/jquery.mjs` directly as there are no restrictions for in-package deep imports if `exports` are not supported; and it's better to avoid definining legacy fields in new setups. Fixes jquerygh-4592
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. The `module` field is removed. It would only be used by legacy bundlers with no `exports` support. Rigth now, this seems to be mostly Parcel <3. However, those bundlers can just import `jquery/dist/jquery.mjs` directly as there are no restrictions for in-package deep imports if `exports` are not supported; and it's better to avoid definining legacy fields in new setups. Fixes jquerygh-4592
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. The `module` field is removed. It would only be used by legacy bundlers with no `exports` support. Rigth now, this seems to be mostly Parcel <3. However, those bundlers can just import `jquery/dist/jquery.mjs` directly as there are no restrictions for in-package deep imports if `exports` are not supported; and it's better to avoid definining legacy fields in new setups. Fixes jquerygh-4592
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. The `module` field is removed. It would only be used by legacy bundlers with no `exports` support. Rigth now, this seems to be mostly Parcel <3. However, those bundlers can just import `jquery/dist/jquery.mjs` directly as there are no restrictions for in-package deep imports if `exports` are not supported; and it's better to avoid definining legacy fields in new setups. Fixes jquerygh-4592
Node.js has an interoperability layer between ESM & CommonJS that makes `module.exports` available as a default export when a CommonJS module is imported from an ECMAScript one. Since we only have a single default export, this means Node.js only needs a single entry point and the whole setup can be drastically simplified. The `module` field is removed. It would only be used by legacy bundlers with no `exports` support. Rigth now, this seems to be mostly Parcel <3. However, those bundlers can just import `jquery/dist/jquery.mjs` directly as there are no restrictions for in-package deep imports if `exports` are not supported; and it's better to avoid definining legacy fields in new setups. Fixes jquerygh-4592
Summary of the changes: * define the `exports` field in `package.json`; `jQuery` & `$` are also exported as named exports in ESM builds now * declare `"type": "module"` globally except for the `build` folder * add the `--esm` option to `grunt custom`, generating jQuery as an ECMAScript module into the `dist-module` folder * expand `node_smoke_tests` to test the slim & ESM builds and their various combinations; also, test both jQuery loaded via a path to the file as well as from module specifiers that should be parsed via the `exports` feature * add details about ESM usage to the release package README * run `compare_size` on all built minified files; don't run it anymore on unminified files where they don't provide lots of value * remove the remove_map_comment task; SWC doesn't insert the `//# sourceMappingURL=` pragma by default so there's nothing to strip Fixes gh-4592 Closes gh-5255
Description
With your ESM work on source, will you provide an ESM distribution file as well? An ESM distribution can be handy for (ESM-based) projects that don't have a compilation step but which wish to avoid polluting HTML with their dependencies (i.e., by using
import jquery from './node_modules/jquery/dist/jquery-esm.js';
or such).And can you point to a non-bundled entry point with
module
exports
inpackage.json
so bundlers like Rollup can discover it, so thatimport jquery from 'jquery'
will work for such projects out of the box, e.g., without need for adding additional Rollup plugins or having to dig through source to find the entry point?There are various references about*For native ESM that works with Node, see https://nodejs.org/api/packages.html *module
as links at https://stackoverflow.com/questions/42708484/what-is-the-module-package-json-field-forThe text was updated successfully, but these errors were encountered: