语法错误 - 当前未启用对实验性语法“decorators-legacy”的支持。

122

我正在尝试使用修饰器来构建JS React项目。我的.babelrc文件如下:

{
  "presets": [
    "@babel/preset-env",
    "@babel/preset-react",

  ],
  "plugins": [
    "@babel/plugin-transform-runtime",
    "@babel/plugin-transform-object-assign",
    [
      "@babel/plugin-proposal-decorators",
      {
        "legacy": true
      }
    ],
    ["@babel/plugin-proposal-class-properties", { "loose": true }]
  ]
}

再次出现添加@babel/plugin-proposal-decorators问题。

我正在使用babel 7,webpack 4和react 16.5

webpack.config.js:

const path = require("path");
const webpack = require("webpack");
const componentName = "reports-desktop";
const publicFolderRelativePath = "../../../../../../public/js";
const ignorePlugin = new webpack.IgnorePlugin(/^\.\/locale$/, /moment$/);

module.exports = {
    entry: './reports-desktop.js'
    ,
    output: {
        path: path.resolve(__dirname, publicFolderRelativePath),
        filename: `${componentName}.js`
    },
    module: {
        rules: [
            {
                test: /\.js$/,
                exclude: /node_modules/,
                use: {
                    loader: "babel-loader"
                }
            }
        ]
    },
    plugins: [
        ignorePlugin
    ]
};

package.json:

{
  "name": "captain",
  "version": "1.0.0",
  "description": "",
  "main": "index.js",
  "scripts": {
    "watch": "webpack -w --mode development --progress --color --display-error-details",
    "build": "webpack --mode production"
  },
  "keywords": [],
  "author": "",
  "license": "ISC",
  "devDependencies": {
    "@babel/core": "^7.0.0",
    "@babel/plugin-proposal-class-properties": "^7.0.0",
    "@babel/plugin-proposal-decorators": "^7.0.0",
    "@babel/plugin-transform-object-assign": "^7.0.0",
    "@babel/plugin-transform-runtime": "^7.0.0",
    "@babel/preset-env": "^7.0.0",
    "@babel/preset-react": "^7.0.0",
    "@babel/preset-stage-1": "^7.0.0",
    "@babel/preset-stage-2": "^7.0.0",
    "babel-loader": "^8.0.2",
    "babel-plugin-transform-decorators": "^6.24.1",
    "react": "^16.5.0",
    "react-dom": "^16.5.0",
    "react-redux": "^5.0.7",
    "react-router-dom": "^4.3.1",
    "redux": "^4.0.0",
    "webpack": "^4.17.3",
    "webpack-cli": "^3.1.0"
  },
  "dependencies": {
    "axios": "^0.18.0",
    "dropzone": "^5.5.1",
    "lodash": "^4.17.10",
    "moment": "^2.22.2",
    "prop-types": "^15.6.2",
    "react-addons-update": "^15.6.2",
    "react-bootstrap": "^0.32.4",
    "react-datetime": "^2.15.0",
    "react-dnd": "^5.0.0",
    "react-dnd-html5-backend": "^5.0.1",
    "react-media": "^1.8.0",
    "react-tooltip": "^3.8.1"
  }
}

我可能是在错误地使用@babel/plugin-proposal-decorators吗?文档中说这应该可以解决我的问题,但问题仍然存在。


你解决了这个问题吗?在我看来配置没问题。根据错误堆栈跟踪,我会调试一下出了什么问题。 - nexuzzz
1
不,我试图将我的项目更新到Babel 7,但由于装饰器的原因,我没有成功。问题在于我导入了带有装饰器的文件,因此我无法完全控制该组件。(我不能触及该组件,因为其他项目正在使用它 - 这很糟糕吗?- 是的) - olga_babic
2
安装npm @babel/plugin-proposal-decorators并在我的webpack配置文件的babel插件部分添加["@babel/plugin-proposal-decorators", { "legacy": true }]就足以解决我的问题。我想知道你的.bablerc是否被某种方式忽略了。一个区别是,我没有.baberc文件,所有的babel选项都在我的webpack中指定(我使用了create-react-app并弹出)。 - Brian Pursley
1
@bpursley的解决方案对我也奏效了。我先运行了npm run eject,然后将新插件添加到"babel": { "plugins": [ <插件名称> ], "presets": ["react-app"] }中。 - Jona
14个回答

122

我遇到了同样的问题,但我通过运行npm install --save-dev @babel/plugin-proposal-decorators并在我的.babelrc文件的插件部分中添加["@babel/plugin-proposal-decorators", { "legacy": true }]来使它能够正常工作。

对我而言,.babelrc文件中的插件部分现在看起来像这样:

"plugins": [
  ["@babel/plugin-proposal-decorators", { "legacy": true }]
]

重启服务器。仅使用"@babel/plugin-proposal-decorators"。 - Deke
嗨,你介意看一下这个类似的问题吗?我正在使用NextJS和Now.sh进行部署:https://stackoverflow.com/questions/54332378/now-sh-build-breaking-due-to-support-for-the-experimental-syntax-decorators-le - Leon Gaban
1
有没有针对plugin-proposal-decorators的非作用域包?我将babel降级到了babel 6,但我需要这个包来使用mobx。我只能找到作用域版本。 - wolfsbane
@wolfsbane,我认为他们没有非作用域的包,但是我目前正在使用plugin-proposal-decorators与mobx。 我认为相关的软件包看起来像这样:"babel-core": "^6.26.3""@babel/plugin-proposal-decorators": "^7.1.2""mobx": "^5.5.1" - Christopher Bradshaw
1
当我将它添加到插件中时,我会得到这个编译错误:ERROR in The "path" argument must be of type string. Received an instance of Array - jzybert
哦,我的错。我把它放在了 .eslintrc 的插件部分,而不是 .babelrc。 - jzybert

19

首先,执行以下命令:

npm install customize-cra react-app-rewired @babel/plugin-proposal-decorators --save

在项目根目录下创建一个新文件config-overrides.js,然后执行以下操作:

const { override, addDecoratorsLegacy } = require('customize-cra');
module.exports = override(
 addDecoratorsLegacy()
 );

此外,编辑你的package.json文件:

"scripts": {
 "start": "react-app-rewired start",
 "build": "react-app-rewired build",
 "test": "react-app-rewired test",
 "eject": "react-app-rewired eject"
  },

然后重新启动。


16

摘自mobxjs。如果你仍然有问题,请参考此处。它对我很有帮助。

用于Babel 7的示例配置,在旧版装饰器模式下:

//.babelrc

{
    "presets": ["@babel/preset-env"],
    "plugins": [
        ["@babel/plugin-proposal-decorators", { "legacy": true }],
        ["@babel/plugin-proposal-class-properties", { "loose": true }]
    ]
}
请注意,插件的顺序很重要,plugin-proposal-decorators 应该是你的插件列表中的第一个插件。
"devDependencies": {
    "@babel/core": "^7.1.0",
    "@babel/plugin-proposal-class-properties": "^7.1.0",
    "@babel/plugin-proposal-decorators": "^7.1.0",
    "@babel/preset-env": "^7.1.0"
}

非遗留模式装饰器(第二阶段)正在进行中,请参见#1732

编辑:已更新配置以显示Babel 7的非beta配置


1
完全不起作用。在 packages.json.babelrc 中都设置了,但没有任何变化。 - Janusz 'Ivellios' Kamieński
你尝试在 babel.config.js 中实现了吗?我在 .babelrc 中遇到了问题。@Janusz'Ivellios'Kamieński - Alex Cory
1
不,我没有这样做,因为没有清晰的说明。只有一些零碎的东西散落在周围,甚至没有完整的例子。React本身很容易开始使用,但是对于普通开发人员来说,切换到webpack/babel/node - 哦,天哪,这里发生的事情太超现实了。无论如何:我通过放弃之前的配置,并使用最近的CRA和库来设置项目来简单解决了这个问题。幸运的是,在我的情况下有效。 - Janusz 'Ivellios' Kamieński

15

我使用 yarn add @babel/plugin-proposal-decorators 解决了这个问题。

然后在 babel.config.jsplugins 部分添加了以下内容:

  [
    require('@babel/plugin-proposal-decorators').default,
    {
      legacy: true
    }
  ],

最后我需要重启我的Webpack开发服务器。


虽然我没有测试过,但像@christopher bradshaw的答案所说,并且根据Babel网站的说法,如果您正在使用.babelrc进行配置,则需要将以下内容添加到"plugins"部分:

  ["@babel/plugin-proposal-decorators", { "legacy": true }]

10
如果您在使用ReactJS与MobX时遇到此错误,请不要启用修饰语法,而是利用MobX的内置decorate实用程序将修饰符应用于您的类/对象。

不要:

import { observable, computed, action } from "mobx";

class Timer {
  @observable start = Date.now();
  @observable current = Date.now();

  @computed
  get elapsedTime() {
    return this.current - this.start + "milliseconds";
  }

  @action
  tick() {
    this.current = Date.now();
  }
}

做:

import { observable, computed, action, decorate } from "mobx";

class Timer {
  start = Date.now();
  current = Date.now();

  get elapsedTime() {
    return this.current - this.start + "milliseconds";
  }

  tick() {
    this.current = Date.now();
  }
}
decorate(Timer, {
  start: observable,
  current: observable,
  elapsedTime: computed,
  tick: action
});

我发现这个答案非常有用! 还在mobx网站上找到更多信息:https://mobx.js.org/best/decorators.html - jjwallace
15
这只是一个解决问题的临时方法,而不是实际的解决方案。选择不使用装饰器是合理的,但如果你决定想在代码中使用装饰器语法,你应该设置好配置让它们正常工作,而不是放弃,因为所需的神奇的Babel咒语过于晦涩难懂。 - Aaron Greenwald
1
@wuliwong的答案可以使Babel发挥作用,这正是我们所追求的:https://dev59.com/7VQK5IYBdhLWcg3wNtPA#55985896 - hacklikecrack
顺便提一下,MobX 6中已经移除了decorate。 - FredyC
1
哇...谢谢你的解释。真的很有帮助。在你的回答之前,我浪费了两天时间来适应ReactJS中的observable,但是在你的回答后,我测试了场景,它可以工作。 - UGandhi

5
  1. 在 package.json 中安装装饰器(decorator)
"@babel/plugin-proposal-decorators": "^7.3.0",
"@babel/preset-flow": "^7.0.0"
  1. 使用以下内容更新babe.config.js文件
module.exports = {
  "presets": [
    "module:metro-react-native-babel-preset",
    "@babel/preset-flow"
  ],
  "plugins": [
    ["@babel/plugin-proposal-decorators", { "legacy" : true }],
    ["@babel/plugin-proposal-class-properties", { "loose" : true }],
  ]
}

4

很遗憾,上面提到的解决方案都不适用于我。因为它们需要你先运行npm run eject,而我不想这样做。为了在运行时更改和覆盖webpack的配置,有一个叫做react-app-rewired的包,应该这样使用:

首先安装所需的依赖:

npm i --save-dev customize-cra react-app-rewired

然后在项目的根文件夹中添加一个名为config-overrides.js的新文件,并将以下内容添加到该文件中,以启用传统装饰器Babel插件:

const {
  override,
  addDecoratorsLegacy,
  disableEsLint
} = require("customize-cra");

module.exports = override(
  // enable legacy decorators babel plugin
  addDecoratorsLegacy(),

  // disable eslint in webpack
  disableEsLint()
);

最后,修改package.json文件以使用react-app-rewired:
  "scripts": {
    "start": "react-app-rewired start",
    "build": "react-app-rewired build",
    "test": "react-app-rewired test",
    "eject": "react-app-rewired eject"
  },

现在像往常一样运行npm start,并享受它!


3

截至2021年,使用Create React App 4.0只需要以下步骤。

  1. 确保您不要弹射

  2. npm i --save-dev customize-cra react-app-rewired

  3. 添加config-overrides.js,其中包含:

const { addDecoratorsLegacy, override } = require('customize-cra')

module.exports = override(addDecoratorsLegacy())
  1. 添加 babel.config.js 文件,其内容如下:
module.exports = {
  plugins: [['@babel/plugin-proposal-decorators', { legacy: true }]],
}
  1. 打开package.json / scripts部分,并将react-scripts替换为react-app-rewired
"scripts": {
  "start": "react-app-rewired start",
  "build": "react-app-rewired build",
  "test": "react-app-rewired test",
}

之后,这个代码库将能够使用startbuildtest命令。


已经解决了我的问题,我不再遇到任何错误了 :) - xing Zì

2

package.json

"@babel/core": "^7.2.2",
"@babel/plugin-proposal-class-properties": "^7.1.0",
"@babel/plugin-proposal-decorators": "^7.1.0",
"@babel/preset-env": "^7.2.3",
"@babel/preset-react": "^7.0.0",
"@babel/register": "^7.0.0",
"babel-loader": "^8.0.5"

webpack.config.js

presets: ["@babel/preset-env", "@babel/preset-react"]

.babelrc

{
  "presets": [
    "@babel/preset-env",
    "@babel/preset-react"
  ],
  "plugins":  [
    ["@babel/plugin-proposal-decorators", { "legacy": true }],
    ["@babel/plugin-proposal-class-properties", { "loose": true }]
  ]
}

2

To use Mobx with Babel in 2020...

1

npm i babel-preset-mobx

2

module.exports = {
  presets: ['other-presets', 'mobx'],
};

因此,只需安装 mobx 预设并将其添加到您的 babel 配置文件中的 presets 数组中。例如,在 babel.config.js 中等。

完美,应该是现在被接受的答案 :) @olga-babic - Karl Adler
1
这个软件包似乎已经被放弃了3年,我不会依赖它。 - vsync

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接