The Publishing Project

CDS 2020: Moving to Modern Javascript

Handling both current and older browsers Javascript has gotten more complicated since Javascript adopted an annual release schedule.

We are not talking about the stark difference between ES2015/ES6 and ES5 but on a more nuanced set of differences and what it means for transpilation and browser support.

Take, for example, Async/Await, a feature introduced in ES2017. If we want to use it in production we have to provide a fallback for browsers that may meet all our other Javascript requirements. Take the following list of Javascript features along with the version of the language they were introduced in:

If we want to support all the items in the list, then we have to support ES2017 for async/await and assume that if the browsers support ES2017 then it’ll support ES2015 features ask well.

With the information above, and for the purpose of this post, we’ll stick with ES2017 as the closest to modern syntax we have today.

We can use the following browserslist configuration as a starting point. We’re basically telling browserslist-aware tools that to execute the assigned task for all browsers older than these versions.

Normally I would have used a yearly preset like preset-es2017 but the Babel team now discourages using those and are deprecated since Babel 6.

  "browserslist": [
    "Firefox < 84",
    "Chrome < 87",
    "Edge < 87",
    "Safari < 13",
    "iOS < 13.7"
  ],

Babel provides two plugins that use these lists are @babel/preset-env and @babel/preset-modules to make your WebPack bundles leaner by only compiling code for browsers on your list. preset-modules does some additional optimizations for modules that are not part of preset-env yet.

Also, see Philip Walton’s Deploying ES2015+ Code in Production Today for examples of WebPack configurations to build both modern and legacy code bundles

Rollup has its own rollup-plugin-babel to handle processing with Babel.

Serving content to different browsers

The best way to serve different files based on javascript support is to use the module/nomodule pattern.

Using type="module" on script tags for modern files allows you to use modules and modern features on your browser code today.

For browsers that don’t support the features that we want, use the nomodule attribute for the legacy files.

<script type="module" src="modern.js"></script>
<script nomodule src="legacy.js"></script>

Browsers that support type=”module” will load modern.js, and legacy browsers will ignore it.

Package exports in Node

In our package.json file, we can now define another “main” script file using the exports key.

{
  "name": "my-package",
  "exports": "./modern.js",
  "main": "./legacy.js"
}

The idea is that if a version of Node can read the exports filed in the package.json file, then it’ll execute that script and if it doesn’t then it’ll execute the script in the main entry point.

This field is only supported in Node 12.8 and above, which implies support for ES2019+ syntax.