Huáng Jùnliàng 28d7442aae Parse async do expressions (#13043)
* parse async do expressions

* add test cases

* update test fixtures

* chore: add syntax-async-do-expressions

* generater support

* fix: do not transform async do expressions

* chore: add asyncDoExpressions to missing plugin helpers

* update ast types

* add more test cases

* throw when asyncDoExpressions is enabled but not doExpressions

* avoid add parentheses for async do expressions

* address review comments

* chore: update parser typings
2021-04-28 18:26:01 +02:00

320 lines
7.3 KiB
TypeScript

import type Printer from "../printer";
import * as t from "@babel/types";
import * as n from "../node";
export function UnaryExpression(this: Printer, node: t.UnaryExpression) {
if (
node.operator === "void" ||
node.operator === "delete" ||
node.operator === "typeof" ||
// throwExpressions
node.operator === "throw"
) {
this.word(node.operator);
this.space();
} else {
this.token(node.operator);
}
this.print(node.argument, node);
}
export function DoExpression(this: Printer, node: t.DoExpression) {
if (node.async) {
this.word("async");
this.space();
}
this.word("do");
this.space();
this.print(node.body, node);
}
export function ParenthesizedExpression(
this: Printer,
node: t.ParenthesizedExpression,
) {
this.token("(");
this.print(node.expression, node);
this.token(")");
}
export function UpdateExpression(this: Printer, node: t.UpdateExpression) {
if (node.prefix) {
this.token(node.operator);
this.print(node.argument, node);
} else {
this.startTerminatorless(true);
this.print(node.argument, node);
this.endTerminatorless();
this.token(node.operator);
}
}
export function ConditionalExpression(
this: Printer,
node: t.ConditionalExpression,
) {
this.print(node.test, node);
this.space();
this.token("?");
this.space();
this.print(node.consequent, node);
this.space();
this.token(":");
this.space();
this.print(node.alternate, node);
}
export function NewExpression(
this: Printer,
node: t.NewExpression,
parent: any,
) {
this.word("new");
this.space();
this.print(node.callee, node);
if (
this.format.minified &&
node.arguments.length === 0 &&
!node.optional &&
!t.isCallExpression(parent, { callee: node }) &&
!t.isMemberExpression(parent) &&
!t.isNewExpression(parent)
) {
return;
}
this.print(node.typeArguments, node); // Flow
this.print(node.typeParameters, node); // TS
if (node.optional) {
this.token("?.");
}
this.token("(");
this.printList(node.arguments, node);
this.token(")");
}
export function SequenceExpression(this: Printer, node: t.SequenceExpression) {
this.printList(node.expressions, node);
}
export function ThisExpression(this: Printer) {
this.word("this");
}
export function Super(this: Printer) {
this.word("super");
}
export function Decorator(this: Printer, node: t.Decorator) {
this.token("@");
this.print(node.expression, node);
this.newline();
}
export function OptionalMemberExpression(
this: Printer,
node: t.OptionalMemberExpression,
) {
this.print(node.object, node);
if (!node.computed && t.isMemberExpression(node.property)) {
throw new TypeError("Got a MemberExpression for MemberExpression property");
}
let computed = node.computed;
// @ts-expect-error todo(flow->ts) maybe instead of typeof check specific literal types?
if (t.isLiteral(node.property) && typeof node.property.value === "number") {
computed = true;
}
if (node.optional) {
this.token("?.");
}
if (computed) {
this.token("[");
this.print(node.property, node);
this.token("]");
} else {
if (!node.optional) {
this.token(".");
}
this.print(node.property, node);
}
}
export function OptionalCallExpression(
this: Printer,
node: t.OptionalCallExpression,
) {
this.print(node.callee, node);
this.print(node.typeArguments, node); // Flow
this.print(node.typeParameters, node); // TS
if (node.optional) {
this.token("?.");
}
this.token("(");
this.printList(node.arguments, node);
this.token(")");
}
export function CallExpression(this: Printer, node: t.CallExpression) {
this.print(node.callee, node);
this.print(node.typeArguments, node); // Flow
this.print(node.typeParameters, node); // TS
this.token("(");
this.printList(node.arguments, node);
this.token(")");
}
export function Import(this: Printer) {
this.word("import");
}
function buildYieldAwait(keyword: string) {
return function (node: any) {
this.word(keyword);
if (node.delegate) {
this.token("*");
}
if (node.argument) {
this.space();
const terminatorState = this.startTerminatorless();
this.print(node.argument, node);
this.endTerminatorless(terminatorState);
}
};
}
export const YieldExpression = buildYieldAwait("yield");
export const AwaitExpression = buildYieldAwait("await");
export function EmptyStatement(this: Printer) {
this.semicolon(true /* force */);
}
export function ExpressionStatement(
this: Printer,
node: t.ExpressionStatement,
) {
this.print(node.expression, node);
this.semicolon();
}
export function AssignmentPattern(this: Printer, node: t.AssignmentPattern) {
this.print(node.left, node);
// @ts-expect-error todo(flow->ts) property present on some of the types in union but not all
if (node.left.optional) this.token("?");
// @ts-expect-error todo(flow->ts) property present on some of the types in union but not all
this.print(node.left.typeAnnotation, node);
this.space();
this.token("=");
this.space();
this.print(node.right, node);
}
export function AssignmentExpression(
this: Printer,
node: t.AssignmentExpression,
parent: any,
) {
// Somewhere inside a for statement `init` node but doesn't usually
// needs a paren except for `in` expressions: `for (a in b ? a : b;;)`
const parens =
this.inForStatementInitCounter &&
node.operator === "in" &&
!n.needsParens(node, parent);
if (parens) {
this.token("(");
}
this.print(node.left, node);
this.space();
if (node.operator === "in" || node.operator === "instanceof") {
this.word(node.operator);
} else {
this.token(node.operator);
}
this.space();
this.print(node.right, node);
if (parens) {
this.token(")");
}
}
export function BindExpression(this: Printer, node: t.BindExpression) {
this.print(node.object, node);
this.token("::");
this.print(node.callee, node);
}
export {
AssignmentExpression as BinaryExpression,
AssignmentExpression as LogicalExpression,
};
export function MemberExpression(this: Printer, node: t.MemberExpression) {
this.print(node.object, node);
if (!node.computed && t.isMemberExpression(node.property)) {
throw new TypeError("Got a MemberExpression for MemberExpression property");
}
let computed = node.computed;
// @ts-expect-error todo(flow->ts) maybe use specific literal types
if (t.isLiteral(node.property) && typeof node.property.value === "number") {
computed = true;
}
if (computed) {
this.token("[");
this.print(node.property, node);
this.token("]");
} else {
this.token(".");
this.print(node.property, node);
}
}
export function MetaProperty(this: Printer, node: t.MetaProperty) {
this.print(node.meta, node);
this.token(".");
this.print(node.property, node);
}
export function PrivateName(this: Printer, node: t.PrivateName) {
this.token("#");
this.print(node.id, node);
}
export function V8IntrinsicIdentifier(
this: Printer,
node: t.V8IntrinsicIdentifier,
) {
this.token("%");
this.word(node.name);
}
export function ModuleExpression(node: t.ModuleExpression) {
this.word("module");
this.space();
this.token("{");
if (node.body.body.length === 0) {
this.token("}");
} else {
this.newline();
this.printSequence(node.body.body, node, { indent: true });
this.rightBrace();
}
}