Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

decorator-transforms

ef4449.6kMIT2.3.0TypeScript support: included

Better babel transforms for decorators

readme

decorator-transforms

Better babel transforms for decorators.

Why?

A transform like @babel/plugin-proposal-decorators is still often necessary because it took Decorators a long time to move through the TC39 standardization process. But we can implement the same functionality in a much simpler way now that we can rely on the presence of features like class static block, private fields, and WeakMap. This results in significantly smaller code and avoids the need to transpile other class features.

Status

This is in use in several large, legacy codebases. As far as I can tell, it implements the complete API surface of ['@babel/plugin-proposal-decorators', { legacy: true }]. Please report any differences, we aim for bug-for-bug compatibility.

Browser Support

Under our default settings, browsers will need to support

  • private fields
  • static blocks
  • WeakMap

If you use the runEarly: true option (see below) in conjunction with an appropriately-configured @babel/preset-env, browsers will only need to support

  • WeakMap

Options

runtime

The runtime option controls how the emitted code will find the (small) runtime helper module. You can set it to:

  • "globals" (default), which means the helpers must be installed in a global variable. You can install them by adding import "decorator-transform/globals" at the very beginning of your app.
  • { import: "some-module-path" }, which means the helpers will be imported as needed from the module path you specify. The module path "decorator-transforms/runtime" is available within this package for this purpose, but keep in mind that it might not always work if you're transpiling third-party dependencies that cannot necessarily resolve your app's dependencies. In that case you might want to require.resolve it to an absolute path instead.

Example Config:

{
  plugins: [
    [
      'decorator-transforms',
      {
        runtime: {
          import: require.resolve('decorator-transforms/runtime'),
        },
      },
    ],
  ];
}

runEarly

By default, decorator-transforms runs like any normal babel plugin. This works fine when you're targeting any browsers that natively support private fields and static blocks.

But if you try to transpile away private fields or static blocks, the fairly aggressive timing of those transforms in @babel/preset-env means that they will run first and

  1. Incorrectly tell you to install @babel/plugin-transform-decorators (which you don't need because you have decorator-transforms).
  2. Fail to transpile-away the private fields and static blocks emitted by decorator-transforms.

The solution to both problems is setting runEarly: true on decorator-transforms. This setting is not the default because it does incur the cost of an extra traversal in babel's pre phase.

A Note on Naming Conventions

If you try to use the string name "decorator-transforms" in a babel config file, babel will assume you mean the NPM package "babel-plugin-decorator-transforms". This is a terrible convention. The solution is to use the string "module:decorator-transforms" instead.

Trying this in an Ember App

  1. Install the decorator-transforms package.
  2. In ember-cli-build.js:

    new EmberApp(defaults, {
      'ember-cli-babel': {
         // turn off the old transform
         // (for this to work when using Embroider you need https://github.com/embroider-build/embroider/pull/1673)
         disableDecoratorTransforms: true,
       },
       babel: {
         plugins: [
           // add the new transform.
           require.resolve('decorator-transforms'),
         ],
       },
    )
  3. At the beginning of app.js, install the global runtime helpers:

    import 'decorator-transforms/globals';

    In classic builds, "globals" is the only runtime setting that works because ember-auto-import cannot see the output of this babel transform.

    In Embroider (post https://github.com/embroider-build/embroider/pull/1673), you can use runtime: { import: require.resolve("decorator-transforms/runtime") } and then you don't need to manually install the globals.

changelog

Changelog

Release (2024-11-07)

decorator-transforms 2.3.0 (minor)

:rocket: Enhancement

Committers: 1

Release (2024-11-05)

decorator-transforms 2.2.4 (patch)

:bug: Bug Fix

  • decorator-transforms
    • #44 Revert "Don't use nodeisms" (@ef4)

Committers: 1

  • Edward Faulkner (@ef4)

Release (2024-11-05)

decorator-transforms 2.2.3 (patch)

:bug: Bug Fix

:house: Internal

Committers: 1

Release (2024-10-01)

decorator-transforms 2.2.2 (patch)

:bug: Bug Fix

  • decorator-transforms
    • #35 Remove unintentional change that was part of #33 (@ef4)

Committers: 1

  • Edward Faulkner (@ef4)

Release (2024-10-01)

decorator-transforms 2.2.1 (patch)

:bug: Bug Fix

  • decorator-transforms

Committers: 1

  • Edward Faulkner (@ef4)

Release (2024-09-19)

decorator-transforms 2.2.0 (minor)

:rocket: Enhancement

  • decorator-transforms

Committers: 1

  • Edward Faulkner (@ef4)

Release (2024-09-17)

decorator-transforms 2.1.0 (minor)

:rocket: Enhancement

  • decorator-transforms
    • #30 Offer an explicit import path for esm vs cjs runtime (@ef4)

Committers: 1

  • Edward Faulkner (@ef4)

Release (2024-04-27)

decorator-transforms 2.0.0 (major)

:boom: Breaking Change

  • decorator-transforms
    • #26 Implement runEarly for arbitrary browser compatibility (@ef4)

:house: Internal

  • decorator-transforms
    • #19 Test interop with template-colocation-plugin (@ef4)

Committers: 1

  • Edward Faulkner (@ef4)

Release (2024-04-16)

decorator-transforms 1.2.1 (patch)

:bug: Bug Fix

  • decorator-transforms
    • #24 add a prepare script to make sure it builds (@mansona)

Committers: 1

Release (2024-04-16)

decorator-transforms 1.2.0 (minor)

:rocket: Enhancement

  • decorator-transforms

:house: Internal

Committers: 2

1.1.0 (2024-01-15)

  • ENHANCEMENT: implement field and method decorators on plain javascript objects
  • BUGFIX: fix naming collisions with existing private fields

1.0.3 (2024-01-04)

  • BUGFIX: v1.0.2 was not runtime-compatible with prior 1.x releases, so mixing versions would cause errors. This restores the stable API between them.

1.0.2 (2024-01-03)

  • BUGFIX: support decorating static fields and static methods

1.0.1 (2023-11-20)

  • BUGFIX: support fields and methods with names that are string literals, numeric literals, bigint literals, or computed expressions
  • BUGFIX: pass initializer: null to field decortors when there is no initial value, for consistency with older implementation.

1.0.0 (2023-11-18)

  • First feature-complete release