import { hasOwnDecorators } from "./decorators"; export const FEATURES = Object.freeze({ //classes: 1 << 0, fields: 1 << 1, privateMethods: 1 << 2, decorators: 1 << 3, }); // We can't use a symbol because this needs to always be the same, even if // this package isn't deduped by npm. e.g. // - node_modules/ // - @babel/plugin-class-features // - @babel/plugin-proposal-decorators // - node_modules // - @babel-plugin-class-features const featuresKey = "@babel/plugin-class-features/featuresKey"; const looseKey = "@babel/plugin-class-features/looseKey"; export function enableFeature(file, feature, loose) { // We can't blindly enable the feature because, if it was already set, // "loose" can't be changed, so that // @babel/plugin-class-properties { loose: true } // @babel/plugin-class-properties { loose: false } // is transformed in loose mode. // We only enabled the feature if it was previously disabled. if (!hasFeature(file, feature)) { file.set(featuresKey, file.get(featuresKey) | feature); if (loose) file.set(looseKey, file.get(looseKey) | feature); } } function hasFeature(file, feature) { return !!(file.get(featuresKey) & feature); } export function isLoose(file, feature) { return !!(file.get(looseKey) & feature); } export function verifyUsedFeatures(path, file) { if (hasOwnDecorators(path.node)) { if (!hasFeature(file, FEATURES.decorators)) { throw path.buildCodeFrameError( "Decorators are not enabled." + "\nIf you are using " + '["@babel/plugin-proposal-decorators", { "legacy": true }], ' + 'make sure it comes *before* "@babel/plugin-proposal-class-properties" ' + "and enable loose mode, like so:\n" + '\t["@babel/plugin-proposal-decorators", { "legacy": true }]\n' + '\t["@babel/plugin-proposal-class-properties", { "loose": true }]', ); } if (path.isPrivate()) { throw path.buildCodeFrameError( `Private ${ path.isClassMethod() ? "methods" : "fields" } in decorated classes are not supported yet.`, ); } } // NOTE: We can't use path.isPrivateMethod() because it isn't supported in <7.2.0 if (path.isPrivate() && path.isMethod()) { if (!hasFeature(file, FEATURES.privateMethods)) { throw path.buildCodeFrameError("Class private methods are not enabled."); } } if ( hasFeature(file, FEATURES.privateMethods) && hasFeature(file, FEATURES.fields) && isLoose(file, FEATURES.privateMethods) !== isLoose(file, FEATURES.fields) ) { throw path.buildCodeFrameError( "'loose' mode configuration must be the same for both @babel/plugin-proposal-class-properties " + "and @babel/plugin-proposal-private-methods", ); } if (path.isProperty()) { if (!hasFeature(file, FEATURES.fields)) { throw path.buildCodeFrameError("Class fields are not enabled."); } } }