ESM and transpiling

To support every modern front end environment, the @dnb/eufemia supports different transpiled module formats:

  • ESM with ES5 (default)
  • ES with ES6
  • CJS with ES5
  • UMD with ES5

Default module format

The @dnb/eufemia uses ESM as the default module format. This allows us to more easily and by default tree shaking support:

// Imports only the code needed for the button
import { Button } from '@dnb/eufemia'
import { Button } from '@dnb/eufemia/es'
// Imports only the code needed for icons
import { question } from '@dnb/eufemia/icons'
import { question } from '@dnb/eufemia/es/icons'

CommonJS (CJS)

Node.js uses RequireJS and has CommonJS as their default module format.

SSR

In case you are using the @dnb/eufemia in a classic Node.js environment, like typical in SSR, you can import or require everything from the /cjs subfolder:

// Components
import { Button } from '@dnb/eufemia/cjs'
const { Button } = require('@dnb/eufemia/cjs/components')
const Button = require('@dnb/eufemia/cjs/components/Button')
// Icons
import { question } from '@dnb/eufemia/cjs/icons'
const { question } = require('@dnb/eufemia/cjs/icons')
const question = require('@dnb/eufemia/cjs/icons/question')
// Styles
import '@dnb/eufemia/cjs/style'
require('@dnb/eufemia/cjs/style')

Next.js and Node.js version 12 and above

NB: If you use Next.js with Node.js version 12 and above, we may have to define explicitly that we want to use CommonJS:

import { Button } from '@dnb/eufemia/cjs'
const { Button } = require('@dnb/eufemia/cjs/components')

Alternatively, you can make use of next-transpile-modules.

ESM in Node.js version 10

You can easily use ESM in Node.js environment. Have a look at the Next.js example.

  1. Install the esm package: npm i esm -D
  2. Call Node with an environment variable: NODE_OPTIONS='-r esm' node ...
// package.json
"start": "NODE_OPTIONS='-r esm' next start ./src"

Jest and ESM (Node testing environments)

Older Jest versions uses still CommonJS as the default module format. If you use the default @dnb/eufemia imports, then you get a mismatch between ES module and CommonJS formats. To ensure that Jest transforms your code in to CSJ, you can use the following Jest configuration --moduleNameMapper

jest --moduleNameMapper '{"@dnb/eufemia(.*)":"@dnb/eufemia/cjs$1"}'

or in a jest.config.js or jest.preset.js file:

module.exports = {
...
moduleNameMapper: { '@dnb/eufemia(.*)': '@dnb/eufemia/cjs$1' }
}

Pure ES6

In some cases you may want to import an ES6 version.

// ES6 version of all Components
import { Button } from '@dnb/eufemia/es'
import { Button } from '@dnb/eufemia/es/components'
// ES6 version of a single component
import Button from '@dnb/eufemia/es/components/Button'

Make sure your environment has a production build with:

  • Tree Shaking
  • Compile to ES5 for IE support
  • Minify and mangle the Output

By default, webpack v4 does this fine.

Tree shaking

To optimize the bundle size, we have several options. It also depends on what is used to make the bundle. Is webpack used, then you may read a good guide on Tree Shaking.

If you only need / import a few components, you may consider to import them directly, without named imports (ES5):

import MyMaskedInput from '@dnb/eufemia/components/InputMasked'
import MyMaskedInput from '@dnb/eufemia/cjs/components/InputMasked'
// or
import Input from '@dnb/eufemia/components/input-masked/InputMasked'
import Input from '@dnb/eufemia/cjs/components/input-masked/InputMasked'

... or as ES6 imports:

import { InputMasked } from '@dnb/eufemia/es/components'