|
|
"use strict";
Object.defineProperty(exports, "__esModule", { value: true }); exports.default = createPlugin;
var _pluginSyntaxJsx = _interopRequireDefault(require("@babel/plugin-syntax-jsx"));
var _helperPluginUtils = require("@babel/helper-plugin-utils");
var _core = require("@babel/core");
var _helperModuleImports = require("@babel/helper-module-imports");
var _helperAnnotateAsPure = _interopRequireDefault(require("@babel/helper-annotate-as-pure"));
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
const DEFAULT = { importSource: "react", runtime: "automatic", pragma: "React.createElement", pragmaFrag: "React.Fragment" }; const JSX_SOURCE_ANNOTATION_REGEX = /\*?\s*@jsxImportSource\s+([^\s]+)/; const JSX_RUNTIME_ANNOTATION_REGEX = /\*?\s*@jsxRuntime\s+([^\s]+)/; const JSX_ANNOTATION_REGEX = /\*?\s*@jsx\s+([^\s]+)/; const JSX_FRAG_ANNOTATION_REGEX = /\*?\s*@jsxFrag\s+([^\s]+)/;
const get = (pass, name) => pass.get(`@babel/plugin-react-jsx/${name}`);
const set = (pass, name, v) => pass.set(`@babel/plugin-react-jsx/${name}`, v);
function createPlugin({ name, development }) { return (0, _helperPluginUtils.declare)((api, options) => { const { pure: PURE_ANNOTATION, throwIfNamespace = true, filter, runtime: RUNTIME_DEFAULT = development ? "automatic" : "classic", importSource: IMPORT_SOURCE_DEFAULT = DEFAULT.importSource, pragma: PRAGMA_DEFAULT = DEFAULT.pragma, pragmaFrag: PRAGMA_FRAG_DEFAULT = DEFAULT.pragmaFrag } = options; { var { useSpread = false, useBuiltIns = false } = options;
if (RUNTIME_DEFAULT === "classic") { if (typeof useSpread !== "boolean") { throw new Error("transform-react-jsx currently only accepts a boolean option for " + "useSpread (defaults to false)"); }
if (typeof useBuiltIns !== "boolean") { throw new Error("transform-react-jsx currently only accepts a boolean option for " + "useBuiltIns (defaults to false)"); }
if (useSpread && useBuiltIns) { throw new Error("transform-react-jsx currently only accepts useBuiltIns or useSpread " + "but not both"); } } } const injectMetaPropertiesVisitor = { JSXOpeningElement(path, state) { for (const attr of path.get("attributes")) { if (!attr.isJSXElement()) continue; const { name } = attr.node.name;
if (name === "__source" || name === "__self") { throw path.buildCodeFrameError(`__source and __self should not be defined in props and are reserved for internal usage.`); } }
const self = _core.types.jsxAttribute(_core.types.jsxIdentifier("__self"), _core.types.jsxExpressionContainer(_core.types.thisExpression()));
const source = _core.types.jsxAttribute(_core.types.jsxIdentifier("__source"), _core.types.jsxExpressionContainer(makeSource(path, state)));
path.pushContainer("attributes", [self, source]); }
}; return { name, inherits: _pluginSyntaxJsx.default, visitor: { JSXNamespacedName(path) { if (throwIfNamespace) { throw path.buildCodeFrameError(`Namespace tags are not supported by default. React's JSX doesn't support namespace tags. \
You can set \`throwIfNamespace: false\` to bypass this warning.`); } },
JSXSpreadChild(path) { throw path.buildCodeFrameError("Spread children are not supported in React."); },
Program: { enter(path, state) { const { file } = state; let runtime = RUNTIME_DEFAULT; let source = IMPORT_SOURCE_DEFAULT; let pragma = PRAGMA_DEFAULT; let pragmaFrag = PRAGMA_FRAG_DEFAULT; let sourceSet = !!options.importSource; let pragmaSet = !!options.pragma; let pragmaFragSet = !!options.pragmaFrag;
if (file.ast.comments) { for (const comment of file.ast.comments) { const sourceMatches = JSX_SOURCE_ANNOTATION_REGEX.exec(comment.value);
if (sourceMatches) { source = sourceMatches[1]; sourceSet = true; }
const runtimeMatches = JSX_RUNTIME_ANNOTATION_REGEX.exec(comment.value);
if (runtimeMatches) { runtime = runtimeMatches[1]; }
const jsxMatches = JSX_ANNOTATION_REGEX.exec(comment.value);
if (jsxMatches) { pragma = jsxMatches[1]; pragmaSet = true; }
const jsxFragMatches = JSX_FRAG_ANNOTATION_REGEX.exec(comment.value);
if (jsxFragMatches) { pragmaFrag = jsxFragMatches[1]; pragmaFragSet = true; } } }
set(state, "runtime", runtime);
if (runtime === "classic") { if (sourceSet) { throw path.buildCodeFrameError(`importSource cannot be set when runtime is classic.`); }
const createElement = toMemberExpression(pragma); const fragment = toMemberExpression(pragmaFrag); set(state, "id/createElement", () => _core.types.cloneNode(createElement)); set(state, "id/fragment", () => _core.types.cloneNode(fragment)); set(state, "defaultPure", pragma === DEFAULT.pragma); } else if (runtime === "automatic") { if (pragmaSet || pragmaFragSet) { throw path.buildCodeFrameError(`pragma and pragmaFrag cannot be set when runtime is automatic.`); }
const define = (name, id) => set(state, name, createImportLazily(state, path, id, source));
define("id/jsx", development ? "jsxDEV" : "jsx"); define("id/jsxs", development ? "jsxDEV" : "jsxs"); define("id/createElement", "createElement"); define("id/fragment", "Fragment"); set(state, "defaultPure", source === DEFAULT.importSource); } else { throw path.buildCodeFrameError(`Runtime must be either "classic" or "automatic".`); }
if (development) { path.traverse(injectMetaPropertiesVisitor, state); } }
}, JSXElement: { exit(path, file) { let callExpr;
if (get(file, "runtime") === "classic" || shouldUseCreateElement(path)) { callExpr = buildCreateElementCall(path, file); } else { callExpr = buildJSXElementCall(path, file); }
path.replaceWith(_core.types.inherits(callExpr, path.node)); }
}, JSXFragment: { exit(path, file) { let callExpr;
if (get(file, "runtime") === "classic") { callExpr = buildCreateElementFragmentCall(path, file); } else { callExpr = buildJSXFragmentCall(path, file); }
path.replaceWith(_core.types.inherits(callExpr, path.node)); }
},
JSXAttribute(path) { if (_core.types.isJSXElement(path.node.value)) { path.node.value = _core.types.jsxExpressionContainer(path.node.value); } }
} };
function call(pass, name, args) { const node = _core.types.callExpression(get(pass, `id/${name}`)(), args);
if (PURE_ANNOTATION != null ? PURE_ANNOTATION : get(pass, "defaultPure")) (0, _helperAnnotateAsPure.default)(node); return node; }
function shouldUseCreateElement(path) { const openingPath = path.get("openingElement"); const attributes = openingPath.node.attributes; let seenPropsSpread = false;
for (let i = 0; i < attributes.length; i++) { const attr = attributes[i];
if (seenPropsSpread && _core.types.isJSXAttribute(attr) && attr.name.name === "key") { return true; } else if (_core.types.isJSXSpreadAttribute(attr)) { seenPropsSpread = true; } }
return false; }
function convertJSXIdentifier(node, parent) { if (_core.types.isJSXIdentifier(node)) { if (node.name === "this" && _core.types.isReferenced(node, parent)) { return _core.types.thisExpression(); } else if (_core.types.isValidIdentifier(node.name, false)) { node.type = "Identifier"; } else { return _core.types.stringLiteral(node.name); } } else if (_core.types.isJSXMemberExpression(node)) { return _core.types.memberExpression(convertJSXIdentifier(node.object, node), convertJSXIdentifier(node.property, node)); } else if (_core.types.isJSXNamespacedName(node)) { return _core.types.stringLiteral(`${node.namespace.name}:${node.name.name}`); }
return node; }
function convertAttributeValue(node) { if (_core.types.isJSXExpressionContainer(node)) { return node.expression; } else { return node; } }
function accumulateAttribute(array, attribute) { if (_core.types.isJSXSpreadAttribute(attribute.node)) { const arg = attribute.node.argument;
if (_core.types.isObjectExpression(arg)) { array.push(...arg.properties); } else { array.push(_core.types.spreadElement(arg)); }
return array; }
const value = convertAttributeValue(attribute.node.name.name !== "key" ? attribute.node.value || _core.types.booleanLiteral(true) : attribute.node.value);
if (attribute.node.name.name === "key" && value === null) { throw attribute.buildCodeFrameError('Please provide an explicit key value. Using "key" as a shorthand for "key={true}" is not allowed.'); }
if (_core.types.isStringLiteral(value) && !_core.types.isJSXExpressionContainer(attribute.node.value)) { var _value$extra;
value.value = value.value.replace(/\n\s+/g, " "); (_value$extra = value.extra) == null ? true : delete _value$extra.raw; }
if (_core.types.isJSXNamespacedName(attribute.node.name)) { attribute.node.name = _core.types.stringLiteral(attribute.node.name.namespace.name + ":" + attribute.node.name.name.name); } else if (_core.types.isValidIdentifier(attribute.node.name.name, false)) { attribute.node.name.type = "Identifier"; } else { attribute.node.name = _core.types.stringLiteral(attribute.node.name.name); }
array.push(_core.types.inherits(_core.types.objectProperty(attribute.node.name, value), attribute.node)); return array; }
function buildChildrenProperty(children) { let childrenNode;
if (children.length === 1) { childrenNode = children[0]; } else if (children.length > 1) { childrenNode = _core.types.arrayExpression(children); } else { return undefined; }
return _core.types.objectProperty(_core.types.identifier("children"), childrenNode); }
function buildJSXElementCall(path, file) { const openingPath = path.get("openingElement"); const args = [getTag(openingPath)]; let attribs = []; const extracted = Object.create(null);
for (const attr of openingPath.get("attributes")) { if (attr.isJSXAttribute() && _core.types.isJSXIdentifier(attr.node.name)) { const { name } = attr.node.name;
switch (name) { case "__source": case "__self": if (extracted[name]) throw sourceSelfError(path, name);
case "key": { const keyValue = convertAttributeValue(attr.node.value);
if (keyValue === null) { throw attr.buildCodeFrameError('Please provide an explicit key value. Using "key" as a shorthand for "key={true}" is not allowed.'); }
extracted[name] = keyValue; break; }
default: attribs.push(attr); } } else { attribs.push(attr); } }
const children = _core.types.react.buildChildren(path.node);
if (attribs.length || children.length) { attribs = buildJSXOpeningElementAttributes(attribs, file, children); } else { attribs = _core.types.objectExpression([]); }
args.push(attribs);
if (development) { var _extracted$key, _extracted$__source, _extracted$__self;
args.push((_extracted$key = extracted.key) != null ? _extracted$key : path.scope.buildUndefinedNode(), _core.types.booleanLiteral(children.length > 1), (_extracted$__source = extracted.__source) != null ? _extracted$__source : path.scope.buildUndefinedNode(), (_extracted$__self = extracted.__self) != null ? _extracted$__self : _core.types.thisExpression()); } else if (extracted.key !== undefined) { args.push(extracted.key); }
return call(file, children.length > 1 ? "jsxs" : "jsx", args); }
function buildJSXOpeningElementAttributes(attribs, file, children) { const props = attribs.reduce(accumulateAttribute, []);
if ((children == null ? void 0 : children.length) > 0) { props.push(buildChildrenProperty(children)); }
return _core.types.objectExpression(props); }
function buildJSXFragmentCall(path, file) { const args = [get(file, "id/fragment")()];
const children = _core.types.react.buildChildren(path.node);
args.push(_core.types.objectExpression(children.length > 0 ? [buildChildrenProperty(children)] : []));
if (development) { args.push(path.scope.buildUndefinedNode(), _core.types.booleanLiteral(children.length > 1)); }
return call(file, children.length > 1 ? "jsxs" : "jsx", args); }
function buildCreateElementFragmentCall(path, file) { if (filter && !filter(path.node, file)) return; return call(file, "createElement", [get(file, "id/fragment")(), _core.types.nullLiteral(), ..._core.types.react.buildChildren(path.node)]); }
function buildCreateElementCall(path, file) { const openingPath = path.get("openingElement"); return call(file, "createElement", [getTag(openingPath), buildCreateElementOpeningElementAttributes(file, path, openingPath.get("attributes")), ..._core.types.react.buildChildren(path.node)]); }
function getTag(openingPath) { const tagExpr = convertJSXIdentifier(openingPath.node.name, openingPath.node); let tagName;
if (_core.types.isIdentifier(tagExpr)) { tagName = tagExpr.name; } else if (_core.types.isLiteral(tagExpr)) { tagName = tagExpr.value; }
if (_core.types.react.isCompatTag(tagName)) { return _core.types.stringLiteral(tagName); } else { return tagExpr; } }
function buildCreateElementOpeningElementAttributes(file, path, attribs) { const runtime = get(file, "runtime"); { if (runtime !== "automatic") { const objs = []; const props = attribs.reduce(accumulateAttribute, []);
if (!useSpread) { let start = 0; props.forEach((prop, i) => { if (_core.types.isSpreadElement(prop)) { if (i > start) { objs.push(_core.types.objectExpression(props.slice(start, i))); }
objs.push(prop.argument); start = i + 1; } });
if (props.length > start) { objs.push(_core.types.objectExpression(props.slice(start))); } } else if (props.length) { objs.push(_core.types.objectExpression(props)); }
if (!objs.length) { return _core.types.nullLiteral(); }
if (objs.length === 1) { return objs[0]; }
if (!_core.types.isObjectExpression(objs[0])) { objs.unshift(_core.types.objectExpression([])); }
const helper = useBuiltIns ? _core.types.memberExpression(_core.types.identifier("Object"), _core.types.identifier("assign")) : file.addHelper("extends"); return _core.types.callExpression(helper, objs); } } const props = []; const found = Object.create(null);
for (const attr of attribs) { const name = _core.types.isJSXAttribute(attr) && _core.types.isJSXIdentifier(attr.name) && attr.name.name;
if (runtime === "automatic" && (name === "__source" || name === "__self")) { if (found[name]) throw sourceSelfError(path, name); found[name] = true; }
accumulateAttribute(props, attr); }
return props.length === 1 && _core.types.isSpreadElement(props[0]) ? props[0].argument : props.length > 0 ? _core.types.objectExpression(props) : _core.types.nullLiteral(); } });
function getSource(source, importName) { switch (importName) { case "Fragment": return `${source}/${development ? "jsx-dev-runtime" : "jsx-runtime"}`;
case "jsxDEV": return `${source}/jsx-dev-runtime`;
case "jsx": case "jsxs": return `${source}/jsx-runtime`;
case "createElement": return source; } }
function createImportLazily(pass, path, importName, source) { return () => { const actualSource = getSource(source, importName);
if ((0, _helperModuleImports.isModule)(path)) { let reference = get(pass, `imports/${importName}`); if (reference) return _core.types.cloneNode(reference); reference = (0, _helperModuleImports.addNamed)(path, importName, actualSource, { importedInterop: "uncompiled", importPosition: "after" }); set(pass, `imports/${importName}`, reference); return reference; } else { let reference = get(pass, `requires/${actualSource}`);
if (reference) { reference = _core.types.cloneNode(reference); } else { reference = (0, _helperModuleImports.addNamespace)(path, actualSource, { importedInterop: "uncompiled" }); set(pass, `requires/${actualSource}`, reference); }
return _core.types.memberExpression(reference, _core.types.identifier(importName)); } }; } }
function toMemberExpression(id) { return id.split(".").map(name => _core.types.identifier(name)).reduce((object, property) => _core.types.memberExpression(object, property)); }
function makeSource(path, state) { const location = path.node.loc;
if (!location) { return path.scope.buildUndefinedNode(); }
if (!state.fileNameIdentifier) { const { filename = "" } = state; const fileNameIdentifier = path.scope.generateUidIdentifier("_jsxFileName"); const scope = path.hub.getScope();
if (scope) { scope.push({ id: fileNameIdentifier, init: _core.types.stringLiteral(filename) }); }
state.fileNameIdentifier = fileNameIdentifier; }
return makeTrace(_core.types.cloneNode(state.fileNameIdentifier), location.start.line, location.start.column); }
function makeTrace(fileNameIdentifier, lineNumber, column0Based) { const fileLineLiteral = lineNumber != null ? _core.types.numericLiteral(lineNumber) : _core.types.nullLiteral(); const fileColumnLiteral = column0Based != null ? _core.types.numericLiteral(column0Based + 1) : _core.types.nullLiteral();
const fileNameProperty = _core.types.objectProperty(_core.types.identifier("fileName"), fileNameIdentifier);
const lineNumberProperty = _core.types.objectProperty(_core.types.identifier("lineNumber"), fileLineLiteral);
const columnNumberProperty = _core.types.objectProperty(_core.types.identifier("columnNumber"), fileColumnLiteral);
return _core.types.objectExpression([fileNameProperty, lineNumberProperty, columnNumberProperty]); }
function sourceSelfError(path, name) { const pluginName = `transform-react-jsx-${name.slice(2)}`; return path.buildCodeFrameError(`Duplicate ${name} prop found. You are most likely using the deprecated ${pluginName} Babel plugin. Both __source and __self are automatically set when using the automatic runtime. Please remove transform-react-jsx-source and transform-react-jsx-self from your Babel config.`); }
|