1307 lines
34 KiB
TypeScript
Raw Normal View History

2018-02-25 00:13:39 +01:00
// Abstract Syntax Tree extras that are not needed in a standalone compiler but quite useful for
// testing the parser.
import {
Token,
Node,
NodeKind,
Source,
CommonTypeNode,
TypeNode,
TypeParameterNode,
SignatureNode,
Expression,
IdentifierExpression,
LiteralExpression,
LiteralKind,
FloatLiteralExpression,
IntegerLiteralExpression,
StringLiteralExpression,
RegexpLiteralExpression,
ArrayLiteralExpression,
AssertionExpression,
AssertionKind,
BinaryExpression,
CallExpression,
CommaExpression,
ElementAccessExpression,
2018-02-27 00:30:04 +01:00
FunctionExpression,
NewExpression,
ParenthesizedExpression,
PropertyAccessExpression,
TernaryExpression,
UnaryPostfixExpression,
UnaryExpression,
UnaryPrefixExpression,
Statement,
BlockStatement,
BreakStatement,
ContinueStatement,
DoStatement,
EmptyStatement,
ExportImportStatement,
ExportStatement,
ExpressionStatement,
ForStatement,
IfStatement,
ImportStatement,
ReturnStatement,
SwitchStatement,
ThrowStatement,
TryStatement,
VariableStatement,
WhileStatement,
ClassDeclaration,
EnumDeclaration,
EnumValueDeclaration,
FieldDeclaration,
FunctionDeclaration,
ImportDeclaration,
InterfaceDeclaration,
MethodDeclaration,
NamespaceDeclaration,
TypeDeclaration,
VariableDeclaration,
DecoratorNode,
ModifierNode,
ModifierKind,
ParameterNode,
ParameterKind,
ExportMember,
SwitchCase,
hasModifier
} from "../ast";
import {
CharCode
} from "../util/charcode";
import { Signature } from "../types";
2018-02-25 00:13:39 +01:00
export function serializeNode(node: Node, sb: string[]): void {
switch (node.kind) {
case NodeKind.SOURCE:
serializeSource(<Source>node, sb);
break;
// types
case NodeKind.TYPE:
serializeTypeNode(<TypeNode>node, sb);
break;
case NodeKind.TYPEPARAMETER:
serializeTypeParameter(<TypeParameterNode>node, sb);
break;
// expressions
case NodeKind.FALSE:
case NodeKind.NULL:
case NodeKind.SUPER:
case NodeKind.THIS:
case NodeKind.TRUE:
case NodeKind.CONSTRUCTOR:
case NodeKind.IDENTIFIER:
serializeIdentifierExpression(<IdentifierExpression>node, sb);
break;
case NodeKind.ASSERTION:
serializeAssertionExpression(<AssertionExpression>node, sb);
break;
case NodeKind.BINARY:
serializeBinaryExpression(<BinaryExpression>node, sb);
break;
case NodeKind.CALL:
serializeCallExpression(<CallExpression>node, sb);
break;
case NodeKind.ELEMENTACCESS:
serializeElementAccessExpression(<ElementAccessExpression>node, sb);
break;
2018-02-27 00:30:04 +01:00
case NodeKind.FUNCTION:
case NodeKind.FUNCTIONARROW:
serializeFunctionExpression(<FunctionExpression>node, sb);
break;
case NodeKind.LITERAL:
serializeLiteralExpression(<LiteralExpression>node, sb);
break;
case NodeKind.NEW:
serializeNewExpression(<NewExpression>node, sb);
break;
case NodeKind.PARENTHESIZED:
serializeParenthesizedExpression(<ParenthesizedExpression>node, sb);
break;
case NodeKind.PROPERTYACCESS:
serializePropertyAccessExpression(<PropertyAccessExpression>node, sb);
break;
case NodeKind.TERNARY:
serializeTernaryExpression(<TernaryExpression>node, sb);
break;
case NodeKind.UNARYPOSTFIX:
serializeUnaryPostfixExpression(<UnaryPostfixExpression>node, sb);
break;
case NodeKind.UNARYPREFIX:
serializeUnaryPrefixExpression(<UnaryPrefixExpression>node, sb);
break;
// statements
case NodeKind.BLOCK:
serializeBlockStatement(<BlockStatement>node, sb);
break;
case NodeKind.BREAK:
serializeBreakStatement(<BreakStatement>node, sb);
break;
case NodeKind.CONTINUE:
serializeContinueStatement(<ContinueStatement>node, sb);
break;
case NodeKind.DO:
serializeDoStatement(<DoStatement>node, sb);
break;
case NodeKind.EMPTY:
serializeEmptyStatement(<EmptyStatement>node, sb);
break;
case NodeKind.EXPORT:
serializeExportStatement(<ExportStatement>node, sb);
break;
case NodeKind.EXPORTIMPORT:
serializeExportImportStatement(<ExportImportStatement>node, sb);
break;
case NodeKind.EXPRESSION:
serializeExpressionStatement(<ExpressionStatement>node, sb);
break;
case NodeKind.FOR:
serializeForStatement(<ForStatement>node, sb);
break;
case NodeKind.IF:
serializeIfStatement(<IfStatement>node, sb);
break;
case NodeKind.IMPORT:
serializeImportStatement(<ImportStatement>node, sb);
break;
case NodeKind.RETURN:
serializeReturnStatement(<ReturnStatement>node, sb);
break;
case NodeKind.SWITCH:
serializeSwitchStatement(<SwitchStatement>node, sb);
break;
case NodeKind.THROW:
serializeThrowStatement(<ThrowStatement>node, sb);
break;
case NodeKind.TRY:
serializeTryStatement(<TryStatement>node, sb);
2018-02-25 00:13:39 +01:00
break;
case NodeKind.VARIABLE:
serializeVariableStatement(<VariableStatement>node, sb);
break;
case NodeKind.WHILE:
serializeWhileStatement(<WhileStatement>node, sb);
break;
// declaration statements
case NodeKind.CLASSDECLARATION:
serializeClassDeclaration(<ClassDeclaration>node, sb);
break;
case NodeKind.ENUMDECLARATION:
serializeEnumDeclaration(<EnumDeclaration>node, sb);
break;
case NodeKind.ENUMVALUEDECLARATION:
serializeEnumValueDeclaration(<EnumValueDeclaration>node, sb);
break;
case NodeKind.FIELDDECLARATION:
serializeFieldDeclaration(<FieldDeclaration>node, sb);
break;
case NodeKind.FUNCTIONDECLARATION:
serializeFunctionDeclaration(<FunctionDeclaration>node, sb);
break;
case NodeKind.IMPORTDECLARATION:
serializeImportDeclaration(<ImportDeclaration>node, sb);
break;
case NodeKind.INTERFACEDECLARATION:
serializeInterfaceDeclaration(<InterfaceDeclaration>node, sb);
break;
case NodeKind.METHODDECLARATION:
serializeMethodDeclaration(<MethodDeclaration>node, sb);
break;
case NodeKind.NAMESPACEDECLARATION:
serializeNamespaceDeclaration(<NamespaceDeclaration>node, sb);
break;
case NodeKind.TYPEDECLARATION:
serializeTypeDeclaration(<TypeDeclaration>node, sb);
break;
case NodeKind.VARIABLEDECLARATION:
serializeVariableDeclaration(<VariableDeclaration>node, sb);
break;
// other
case NodeKind.DECORATOR:
serializeDecorator(<DecoratorNode>node, sb);
break;
case NodeKind.EXPORTMEMBER:
serializeExportMember(<ExportMember>node, sb);
break;
case NodeKind.MODIFIER:
serializeModifier(<ModifierNode>node, sb);
break;
case NodeKind.PARAMETER:
serializeParameter(<ParameterNode>node, sb);
break;
case NodeKind.SWITCHCASE:
serializeSwitchCase(<SwitchCase>node, sb);
break;
default:
assert(false);
break;
}
}
2018-02-25 00:13:39 +01:00
export function serializeSource(source: Source, sb: string[]): void {
for (var i = 0, k = source.statements.length; i < k; ++i) {
2018-02-25 00:13:39 +01:00
serializeTerminatedStatement(source.statements[i], sb);
}
}
// types
export function serializeTypeNode(node: CommonTypeNode, sb: string[]): void {
if (node.kind == NodeKind.SIGNATURE) {
return serializeSignatureNode(<SignatureNode>node, sb);
}
var typeNode = <TypeNode>node;
serializeIdentifierExpression(<IdentifierExpression>typeNode.name, sb);
if (typeNode.typeArguments) {
var k = typeNode.typeArguments.length;
if (k) {
sb.push("<");
serializeTypeNode(typeNode.typeArguments[0], sb);
for (var i = 1; i < k; ++i) {
sb.push(", ");
serializeTypeNode(typeNode.typeArguments[i], sb);
}
sb.push(">");
}
if (node.isNullable) sb.push(" | null");
}
}
export function serializeTypeParameter(node: TypeParameterNode, sb: string[]): void {
serializeIdentifierExpression(node.name, sb);
var extendsType = node.extendsType;
if (extendsType) {
sb.push(" extends ");
serializeTypeNode(extendsType, sb);
}
}
export function serializeSignatureNode(node: SignatureNode, sb: string[]): void {
var isNullable = node.isNullable;
sb.push(isNullable ? "((" : "(");
var explicitThisType = node.explicitThisType;
if (explicitThisType) {
sb.push("this: ");
serializeTypeNode(explicitThisType, sb);
}
var parameters = node.parameterTypes;
var numParameters = parameters.length;
if (numParameters) {
if (explicitThisType) sb.push(", ");
serializeParameter(parameters[0], sb);
for (let i = 1; i < numParameters; ++i) {
sb.push(", ");
serializeParameter(parameters[i], sb);
}
}
var returnType = node.returnType;
if (returnType) {
sb.push(") => ");
serializeTypeNode(returnType, sb);
} else {
sb.push(") => void");
}
if (isNullable) sb.push(") | null");
}
// expressions
2018-02-25 00:13:39 +01:00
export function serializeExpression(node: Expression, sb: string[]): void {
serializeNode(node, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeIdentifierExpression(node: IdentifierExpression, sb: string[]): void {
sb.push(node.text);
}
2018-02-25 00:13:39 +01:00
export function serializeArrayLiteralExpression(node: ArrayLiteralExpression, sb: string[]): void {
sb.push("[");
2018-02-25 00:13:39 +01:00
var elements = node.elementExpressions;
var k = elements.length;
if (k) {
if (elements[0]) serializeExpression(<Expression>elements[0], sb);
for (var i = 1; i < k; ++i) {
sb.push(", ");
2018-02-25 00:13:39 +01:00
if (elements[i]) serializeExpression(<Expression>elements[i], sb);
}
}
sb.push("]");
}
2018-02-25 00:13:39 +01:00
export function serializeAssertionExpression(node: AssertionExpression, sb: string[]): void {
if (node.assertionKind == AssertionKind.PREFIX) {
sb.push("<");
serializeTypeNode(node.toType, sb);
sb.push(">");
serializeExpression(node.expression, sb);
} else {
serializeExpression(node.expression, sb);
sb.push(" as ");
serializeTypeNode(node.toType, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeBinaryExpression(node: BinaryExpression, sb: string[]): void {
serializeExpression(node.left, sb);
sb.push(" ");
sb.push(Token.operatorToString(node.operator));
sb.push(" ");
serializeExpression(node.right, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeCallExpression(node: CallExpression, sb: string[]): void {
serializeExpression(node.expression, sb);
2018-02-25 00:13:39 +01:00
var i: i32, k: i32;
if (node.typeArguments && (k = node.typeArguments.length)) {
sb.push("<");
2018-02-25 00:13:39 +01:00
serializeTypeNode(node.typeArguments[0], sb);
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeTypeNode(node.typeArguments[i], sb);
}
sb.push(">(");
2018-02-25 00:13:39 +01:00
} else {
sb.push("(");
2018-02-25 00:13:39 +01:00
}
if (k = node.arguments.length) {
serializeExpression(node.arguments[0], sb);
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeExpression(node.arguments[i], sb);
2018-02-25 00:13:39 +01:00
}
}
sb.push(")");
}
2018-02-25 00:13:39 +01:00
export function serializeCommaExpression(node: CommaExpression, sb: string[]): void {
var k = assert(node.expressions.length);
serializeExpression(node.expressions[0], sb);
2018-02-25 00:13:39 +01:00
for (var i = 1; i < k; ++i) {
sb.push(",");
serializeExpression(node.expressions[i], sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeElementAccessExpression(node: ElementAccessExpression, sb: string[]): void {
serializeExpression(node.expression, sb);
sb.push("[");
serializeExpression(node.elementExpression, sb);
sb.push("]");
}
2018-02-27 00:30:04 +01:00
export function serializeFunctionExpression(node: FunctionExpression, sb: string[]): void {
var declaration = node.declaration;
var isArrow = node.kind == NodeKind.FUNCTIONARROW;
if (!isArrow) {
if (declaration.name.text.length) {
sb.push("function ");
} else {
sb.push("function");
}
} else {
assert(declaration.name.text.length == 0);
}
serializeFunctionCommon(declaration, sb, isArrow);
}
2018-02-25 00:13:39 +01:00
export function serializeLiteralExpression(node: LiteralExpression, sb: string[]): void {
switch (node.literalKind) {
case LiteralKind.FLOAT:
serializeFloatLiteralExpression(<FloatLiteralExpression>node, sb);
break;
case LiteralKind.INTEGER:
serializeIntegerLiteralExpression(<IntegerLiteralExpression>node, sb);
break;
case LiteralKind.STRING:
serializeStringLiteralExpression(<StringLiteralExpression>node, sb);
break;
case LiteralKind.REGEXP:
serializeRegexpLiteralExpression(<RegexpLiteralExpression>node, sb);
break;
case LiteralKind.ARRAY:
serializeArrayLiteralExpression(<ArrayLiteralExpression>node, sb);
break;
// case LiteralKind.OBJECT:
// serializeObjectLiteralExpression(<ObjectLiteralExpression>node, sb);
// break;
default:
assert(false);
break;
}
}
2018-02-25 00:13:39 +01:00
export function serializeFloatLiteralExpression(node: FloatLiteralExpression, sb: string[]): void {
sb.push(node.value.toString(10));
}
2018-02-25 00:13:39 +01:00
export function serializeIntegerLiteralExpression(node: IntegerLiteralExpression, sb: string[]): void {
sb.push(node.value.toString());
}
2018-02-25 00:13:39 +01:00
export function serializeStringLiteral(str: string, sb: string[], singleQuoted: bool = false): void {
var off = 0;
2018-02-25 00:13:39 +01:00
var quote = singleQuoted ? "'" : "\"";
sb.push(quote);
for (var i = 0, k = str.length; i < k;) {
switch (str.charCodeAt(i)) {
case CharCode.NULL:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, off = i + 1));
sb.push("\\0");
off = ++i;
break;
case CharCode.BACKSPACE:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
off = ++i;
sb.push("\\b");
break;
case CharCode.TAB:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
off = ++i;
sb.push("\\t");
break;
case CharCode.LINEFEED:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
off = ++i;
sb.push("\\n");
break;
case CharCode.VERTICALTAB:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
off = ++i;
sb.push("\\v");
break;
case CharCode.FORMFEED:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
off = ++i;
sb.push("\\f");
break;
case CharCode.CARRIAGERETURN:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
sb.push("\\r");
off = ++i;
break;
case CharCode.DOUBLEQUOTE:
if (!singleQuoted) {
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
sb.push("\\\"");
off = ++i;
2018-02-25 00:13:39 +01:00
} else {
++i;
2018-02-25 00:13:39 +01:00
}
break;
case CharCode.SINGLEQUOTE:
if (singleQuoted) {
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
sb.push("\\'");
off = ++i;
2018-02-25 00:13:39 +01:00
} else {
++i;
2018-02-25 00:13:39 +01:00
}
break;
case CharCode.BACKSLASH:
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
sb.push("\\\\");
off = ++i;
break;
default:
++i;
break;
}
}
2018-02-25 00:13:39 +01:00
if (i > off) sb.push(str.substring(off, i));
sb.push(quote);
}
2018-02-25 00:13:39 +01:00
export function serializeStringLiteralExpression(node: StringLiteralExpression, sb: string[]): void {
serializeStringLiteral(node.value, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeRegexpLiteralExpression(node: RegexpLiteralExpression, sb: string[]): void {
sb.push("/");
sb.push(node.pattern);
sb.push("/");
sb.push(node.patternFlags);
}
2018-02-25 00:13:39 +01:00
export function serializeNewExpression(node: NewExpression, sb: string[]): void {
sb.push("new ");
serializeCallExpression(node, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeParenthesizedExpression(node: ParenthesizedExpression, sb: string[]): void {
sb.push("(");
serializeExpression(node.expression, sb);
sb.push(")");
}
2018-02-25 00:13:39 +01:00
export function serializePropertyAccessExpression(node: PropertyAccessExpression, sb: string[]): void {
serializeExpression(node.expression, sb);
sb.push(".");
serializeIdentifierExpression(node.property, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeTernaryExpression(node: TernaryExpression, sb: string[]): void {
serializeExpression(node.condition, sb);
sb.push(" ? ");
serializeExpression(node.ifThen, sb);
sb.push(" : ");
serializeExpression(node.ifElse, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeUnaryExpression(node: UnaryExpression, sb: string[]): void {
switch (node.kind) {
case NodeKind.UNARYPOSTFIX:
serializeUnaryPostfixExpression(<UnaryPostfixExpression>node, sb);
break;
case NodeKind.UNARYPREFIX:
serializeUnaryPrefixExpression(<UnaryPrefixExpression>node, sb);
break;
default:
assert(false);
break;
}
}
2018-02-25 00:13:39 +01:00
export function serializeUnaryPostfixExpression(node: UnaryPostfixExpression, sb: string[]): void {
serializeExpression(node.operand, sb);
sb.push(Token.operatorToString(node.operator));
}
2018-02-25 00:13:39 +01:00
export function serializeUnaryPrefixExpression(node: UnaryPrefixExpression, sb: string[]): void {
sb.push(Token.operatorToString(node.operator));
serializeExpression(node.operand, sb);
}
// statements
2018-02-25 00:13:39 +01:00
export function serializeStatement(node: Statement, sb: string[]): void {
serializeNode(node, sb);
}
2018-02-25 00:13:39 +01:00
function serializeTerminatedStatement(statement: Statement, sb: string[]): void {
serializeStatement(statement, sb);
2018-02-27 00:30:04 +01:00
if (
!sb.length || // leading EmptyStatement
statement.kind == NodeKind.VARIABLE || // potentially assigns a FunctionExpression
statement.kind == NodeKind.EXPRESSION // potentially assigns a FunctionExpression
) {
sb.push(";\n");
} else {
2018-02-25 00:13:39 +01:00
var last = sb[sb.length - 1];
if (last.length && last.charCodeAt(last.length - 1) == CharCode.CLOSEBRACE) {
sb.push("\n");
2018-02-25 00:13:39 +01:00
} else {
sb.push(";\n");
2018-02-25 00:13:39 +01:00
}
}
}
export function serializeBlockStatement(node: BlockStatement, sb: string[]): void {
sb.push("{\n");
for (var i = 0, k = node.statements.length; i < k; ++i) {
serializeTerminatedStatement(node.statements[i], sb);
}
sb.push("}");
}
2018-02-25 00:13:39 +01:00
export function serializeBreakStatement(node: BreakStatement, sb: string[]): void {
if (node.label) {
sb.push("break ");
serializeIdentifierExpression(node.label, sb);
2018-02-25 00:13:39 +01:00
} else {
sb.push("break");
2018-02-25 00:13:39 +01:00
}
}
2018-02-25 00:13:39 +01:00
export function serializeContinueStatement(node: ContinueStatement, sb: string[]): void {
if (node.label) {
sb.push("continue ");
serializeIdentifierExpression(node.label, sb);
2018-02-25 00:13:39 +01:00
} else {
sb.push("continue");
2018-02-25 00:13:39 +01:00
}
}
2018-02-25 00:13:39 +01:00
export function serializeClassDeclaration(node: ClassDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
sb.push("class ");
2018-02-25 00:13:39 +01:00
serializeIdentifierExpression(node.name, sb);
if (k = node.typeParameters.length) {
sb.push("<");
2018-02-25 00:13:39 +01:00
serializeTypeParameter(node.typeParameters[0], sb);
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeTypeParameter(node.typeParameters[i], sb);
}
sb.push(">");
}
if (node.extendsType) {
sb.push(" extends ");
serializeTypeNode(node.extendsType, sb);
}
2018-02-25 00:13:39 +01:00
if (k = node.implementsTypes.length) {
sb.push(" implements ");
2018-02-25 00:13:39 +01:00
serializeTypeNode(node.implementsTypes[0], sb);
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeTypeNode(node.implementsTypes[i], sb);
}
}
sb.push(" {\n");
for (i = 0, k = node.members.length; i < k; ++i) {
2018-02-25 00:13:39 +01:00
serializeTerminatedStatement(node.members[i], sb);
}
sb.push("}");
}
2018-02-25 00:13:39 +01:00
export function serializeDoStatement(node: DoStatement, sb: string[]): void {
sb.push("do ");
serializeStatement(node.statement, sb);
2018-02-25 00:13:39 +01:00
if (node.statement.kind == NodeKind.BLOCK) {
sb.push(" while (");
2018-02-25 00:13:39 +01:00
} else {
sb.push(";\nwhile (");
2018-02-25 00:13:39 +01:00
}
serializeExpression(node.condition, sb);
sb.push(")");
}
2018-02-25 00:13:39 +01:00
export function serializeEmptyStatement(node: EmptyStatement, sb: string[]): void {
}
export function serializeEnumDeclaration(node: EnumDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
sb.push("enum ");
serializeIdentifierExpression(node.name, sb);
2018-02-25 00:13:39 +01:00
if (k = node.values.length) {
sb.push(" {\n");
serializeEnumValueDeclaration(node.values[0], sb);
for (i = 1; i < k; ++i) {
sb.push(",\n");
2018-02-25 00:13:39 +01:00
serializeEnumValueDeclaration(node.values[i], sb);
}
sb.push("\n}");
} else {
sb.push(" {\n}");
}
}
2018-02-25 00:13:39 +01:00
export function serializeEnumValueDeclaration(node: EnumValueDeclaration, sb: string[]): void {
serializeIdentifierExpression(node.name, sb);
if (node.value) {
sb.push(" = ");
serializeExpression(node.value, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeExportImportStatement(node: ExportImportStatement, sb: string[]): void {
sb.push("export import ");
serializeIdentifierExpression(node.externalName, sb);
sb.push(" = ");
serializeIdentifierExpression(node.name, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeExportMember(node: ExportMember, sb: string[]): void {
serializeIdentifierExpression(node.name, sb);
if (node.externalName.text != node.name.text) {
sb.push(" as ");
serializeIdentifierExpression(node.externalName, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeExportStatement(node: ExportStatement, sb: string[]): void {
var i: i32, k: i32;
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
if (k = node.members.length) {
sb.push("export {\n");
serializeExportMember(node.members[0], sb);
for (i = 1; i < k; ++i) {
sb.push(",\n");
serializeExportMember(node.members[i], sb);
2018-02-25 00:13:39 +01:00
}
sb.push("\n}");
} else {
sb.push("export {}");
}
if (node.path) {
2018-02-25 00:13:39 +01:00
sb.push(" from ");
serializeStringLiteralExpression(node.path, sb);
2018-02-25 00:13:39 +01:00
}
}
2018-02-25 00:13:39 +01:00
export function serializeExpressionStatement(node: ExpressionStatement, sb: string[]): void {
serializeExpression(node.expression, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeFieldDeclaration(node: FieldDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
serializeIdentifierExpression(node.name, sb);
if (node.type) {
sb.push(": ");
serializeTypeNode(node.type, sb);
}
if (node.initializer) {
sb.push(" = ");
serializeExpression(node.initializer, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeForStatement(node: ForStatement, sb: string[]): void {
sb.push("for (");
2018-02-25 00:13:39 +01:00
if (node.initializer) {
serializeStatement(node.initializer, sb);
2018-02-25 00:13:39 +01:00
}
if (node.condition) {
sb.push("; ");
serializeExpression(node.condition, sb);
2018-02-25 00:13:39 +01:00
} else {
sb.push(";");
2018-02-25 00:13:39 +01:00
}
if (node.incrementor) {
sb.push("; ");
serializeExpression(node.incrementor, sb);
2018-02-25 00:13:39 +01:00
} else {
sb.push(";");
2018-02-25 00:13:39 +01:00
}
sb.push(") ");
serializeStatement(node.statement, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeFunctionDeclaration(node: FunctionDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
2018-02-27 00:30:04 +01:00
if (node.name.text.length) {
sb.push("function ");
} else {
sb.push("function");
}
serializeFunctionCommon(node, sb);
}
2018-02-27 00:30:04 +01:00
function serializeFunctionCommon(node: FunctionDeclaration, sb: string[], isArrow: bool = false): void {
2018-02-25 00:13:39 +01:00
var i: i32, k: i32;
serializeIdentifierExpression(node.name, sb);
var signature = node.signature;
if (node.typeParameters) {
if (k = node.typeParameters.length) {
sb.push("<");
serializeTypeParameter(node.typeParameters[0], sb);
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeTypeParameter(node.typeParameters[i], sb);
}
sb.push(">");
}
}
sb.push("(");
if (k = signature.parameterTypes.length) {
serializeParameter(signature.parameterTypes[0], sb);
2018-02-25 00:13:39 +01:00
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeParameter(signature.parameterTypes[i], sb);
2018-02-25 00:13:39 +01:00
}
}
2018-02-27 00:30:04 +01:00
if (isArrow) {
if (node.body) {
if (signature.returnType) {
2018-02-27 00:30:04 +01:00
sb.push("): ");
serializeTypeNode(signature.returnType, sb);
2018-02-27 00:30:04 +01:00
}
sb.push(" => ");
serializeStatement(node.body, sb);
2018-02-27 00:30:04 +01:00
} else {
if (signature.returnType) {
2018-02-27 00:30:04 +01:00
sb.push(" => ");
serializeTypeNode(signature.returnType, sb);
2018-02-27 00:30:04 +01:00
} else {
sb.push(" => void");
}
}
2018-02-25 00:13:39 +01:00
} else {
if (signature.returnType && !hasModifier(ModifierKind.SET, node.modifiers)) {
2018-02-27 00:30:04 +01:00
sb.push("): ");
serializeTypeNode(signature.returnType, sb);
2018-02-27 00:30:04 +01:00
} else {
// TODO: constructor?
2018-02-27 00:30:04 +01:00
sb.push(")");
}
if (node.body) {
sb.push(" ");
serializeStatement(node.body, sb);
}
}
}
2018-02-25 00:13:39 +01:00
export function serializeIfStatement(node: IfStatement, sb: string[]): void {
sb.push("if (");
serializeExpression(node.condition, sb);
sb.push(") ");
serializeStatement(node.ifTrue, sb);
2018-02-25 00:13:39 +01:00
if (node.ifTrue.kind != NodeKind.BLOCK) {
sb.push(";\n");
2018-02-25 00:13:39 +01:00
}
if (node.ifFalse) {
2018-02-25 00:13:39 +01:00
if (node.ifTrue.kind == NodeKind.BLOCK) {
sb.push(" else ");
2018-02-25 00:13:39 +01:00
} else {
sb.push("else ");
2018-02-25 00:13:39 +01:00
}
serializeStatement(node.ifFalse, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeImportDeclaration(node: ImportDeclaration, sb: string[]): void {
serializeIdentifierExpression(node.externalName, sb);
if (node.externalName.text != node.name.text) {
sb.push(" as ");
serializeIdentifierExpression(node.name, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeImportStatement(node: ImportStatement, sb: string[]): void {
sb.push("import ");
if (node.declarations) {
2018-02-25 00:13:39 +01:00
var k = node.declarations.length;
if (k) {
sb.push("{\n");
serializeImportDeclaration(node.declarations[0], sb);
for (var i = 1; i < k; ++i) {
sb.push(",\n");
serializeImportDeclaration(node.declarations[i], sb);
2018-02-25 00:13:39 +01:00
}
sb.push("\n} from ");
} else {
sb.push("{} from ");
}
} else if (node.namespaceName) {
sb.push("* as ");
serializeIdentifierExpression(node.namespaceName, sb);
sb.push(" from ");
}
serializeStringLiteralExpression(node.path, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeInterfaceDeclaration(node: InterfaceDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
sb.push("interface ");
serializeIdentifierExpression(node.name, sb);
2018-02-25 00:13:39 +01:00
if (k = node.typeParameters.length) {
sb.push("<");
2018-02-25 00:13:39 +01:00
serializeTypeParameter(node.typeParameters[0], sb);
for (i = 0; i < k; ++i) {
sb.push(", ");
serializeTypeParameter(node.typeParameters[i], sb);
}
sb.push(">");
}
if (node.extendsType) {
sb.push(" extends ");
serializeTypeNode(node.extendsType, sb);
}
sb.push(" {\n");
for (i = 0, k = node.members.length; i < k; ++i) {
2018-02-25 00:13:39 +01:00
serializeTerminatedStatement(node.members[i], sb);
}
sb.push("}");
}
2018-02-25 00:13:39 +01:00
export function serializeMethodDeclaration(node: MethodDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
serializeFunctionCommon(node, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeNamespaceDeclaration(node: NamespaceDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
sb.push("namespace ");
serializeIdentifierExpression(node.name, sb);
sb.push(" {\n");
for (i = 0, k = node.members.length; i < k; ++i) {
2018-02-25 00:13:39 +01:00
serializeTerminatedStatement(node.members[i], sb);
}
sb.push("}");
}
2018-02-25 00:13:39 +01:00
export function serializeReturnStatement(node: ReturnStatement, sb: string[]): void {
if (node.value) {
sb.push("return ");
serializeExpression(node.value, sb);
2018-02-25 00:13:39 +01:00
} else {
sb.push("return");
2018-02-25 00:13:39 +01:00
}
}
2018-02-25 00:13:39 +01:00
export function serializeSwitchCase(node: SwitchCase, sb: string[]): void {
if (node.label) {
sb.push("case ");
serializeExpression(node.label, sb);
sb.push(":\n");
2018-02-25 00:13:39 +01:00
} else {
sb.push("default:\n");
2018-02-25 00:13:39 +01:00
}
var k = node.statements.length;
if (k) {
serializeTerminatedStatement(node.statements[0], sb);
for (var i = 1; i < k; ++i) {
sb.push("\n");
2018-02-25 00:13:39 +01:00
serializeTerminatedStatement(node.statements[i], sb);
}
}
}
2018-02-25 00:13:39 +01:00
export function serializeSwitchStatement(node: SwitchStatement, sb: string[]): void {
sb.push("switch (");
serializeExpression(node.condition, sb);
sb.push(") {\n");
for (var i = 0, k = node.cases.length; i < k; ++i) {
serializeSwitchCase(node.cases[i], sb);
sb.push("\n");
}
sb.push("}");
}
2018-02-25 00:13:39 +01:00
export function serializeThrowStatement(node: ThrowStatement, sb: string[]): void {
sb.push("throw ");
serializeExpression(node.value, sb);
sb.push(";");
}
2018-02-25 00:13:39 +01:00
export function serializeTryStatement(node: TryStatement, sb: string[]): void {
var i: i32, k: i32;
sb.push("try {\n");
2018-02-25 00:13:39 +01:00
for (i = 0, k = node.statements.length; i < k; ++i) {
serializeStatement(node.statements[i], sb);
sb.push(";\n");
}
if (node.catchVariable) {
sb.push("} catch (");
serializeIdentifierExpression(node.catchVariable, sb);
sb.push(") {\n");
2018-02-25 00:13:39 +01:00
if (node.catchStatements) {
for (i = 0, k = node.catchStatements.length; i < k; ++i) {
serializeStatement(node.catchStatements[i], sb);
sb.push(";\n");
}
2018-02-25 00:13:39 +01:00
}
}
if (node.finallyStatements) {
sb.push("} finally {\n");
for (i = 0, k = node.finallyStatements.length; i < k; ++i) {
serializeStatement(node.finallyStatements[i], sb);
sb.push(";\n");
}
}
sb.push("}");
}
2018-02-25 00:13:39 +01:00
export function serializeTypeDeclaration(node: TypeDeclaration, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
serializeModifier(node.modifiers[i], sb);
sb.push(" ");
}
2018-02-25 00:13:39 +01:00
}
sb.push("type ");
serializeIdentifierExpression(node.name, sb);
sb.push(" = ");
serializeTypeNode(node.alias, sb);
}
2018-02-25 00:13:39 +01:00
export function serializeVariableDeclaration(node: VariableDeclaration, sb: string[]): void {
serializeIdentifierExpression(node.name, sb);
if (node.type) {
sb.push(": ");
serializeTypeNode(node.type, sb);
}
if (node.initializer) {
sb.push(" = ");
serializeExpression(node.initializer, sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeVariableStatement(node: VariableStatement, sb: string[]): void {
var i: i32, k: i32;
if (node.decorators) {
for (i = 0, k = node.decorators.length; i < k; ++i) {
serializeDecorator(node.decorators[i], sb);
sb.push("\n");
}
2018-02-25 00:13:39 +01:00
}
var isConst = false, isLet = false;
2018-02-25 00:13:39 +01:00
if (node.modifiers) {
for (i = 0, k = node.modifiers.length; i < k; ++i) {
var modifier = node.modifiers[i];
switch (modifier.modifierKind) {
case ModifierKind.CONST:
assert(!isLet);
isConst = true;
break;
case ModifierKind.LET:
assert(!isConst);
isLet = true;
break;
default:
serializeModifier(modifier, sb);
sb.push(" ");
break;
}
}
2018-02-25 00:13:39 +01:00
}
sb.push(isConst ? "const " : isLet ? "let " : "var ");
2018-02-25 00:13:39 +01:00
k = assert(node.declarations.length);
serializeVariableDeclaration(node.declarations[0], sb);
for (i = 1; i < k; ++i) {
sb.push(", ");
serializeVariableDeclaration(node.declarations[i], sb);
}
}
2018-02-25 00:13:39 +01:00
export function serializeWhileStatement(node: WhileStatement, sb: string[]): void {
sb.push("while (");
serializeExpression(node.condition, sb);
sb.push(") ");
serializeStatement(node.statement, sb);
}
// other
export function serializeDecorator(node: DecoratorNode, sb: string[]): void {
sb.push("@");
serializeExpression(node.name, sb);
if (node.arguments) {
sb.push("(");
2018-02-25 00:13:39 +01:00
var k = node.arguments.length;
if (k) {
serializeExpression(node.arguments[0], sb);
for (var i = 1; i < k; ++i) {
sb.push(", ");
2018-02-25 00:13:39 +01:00
serializeExpression(node.arguments[i], sb);
}
}
sb.push(")");
}
}
export function serializeModifier(node: ModifierNode, sb: string[]): void {
sb.push(modifierToString(node));
}
export function serializeParameter(node: ParameterNode, sb: string[]): void {
2018-02-25 00:13:39 +01:00
if (node.parameterKind == ParameterKind.REST) {
sb.push("...");
2018-02-25 00:13:39 +01:00
}
serializeIdentifierExpression(node.name, sb);
if (node.type) {
2018-02-25 00:13:39 +01:00
if (node.parameterKind == ParameterKind.OPTIONAL && !node.initializer) {
sb.push("?: ");
2018-02-25 00:13:39 +01:00
} else {
sb.push(": ");
2018-02-25 00:13:39 +01:00
}
serializeTypeNode(node.type, sb);
}
if (node.initializer) {
sb.push(" = ");
serializeExpression(node.initializer, sb);
}
}
// helpers
export function modifierToString(node: ModifierNode): string {
switch (node.modifierKind) {
case ModifierKind.ASYNC: return "async";
case ModifierKind.CONST: return "const";
case ModifierKind.LET: return "let"; // unused
case ModifierKind.DECLARE: return "declare";
case ModifierKind.EXPORT: return "export";
case ModifierKind.IMPORT: return "import";
case ModifierKind.STATIC: return "static";
case ModifierKind.ABSTRACT: return "abstract";
case ModifierKind.PUBLIC: return "public";
case ModifierKind.PRIVATE: return "private";
case ModifierKind.PROTECTED: return "protected";
case ModifierKind.READONLY: return "readonly";
case ModifierKind.GET: return "get";
case ModifierKind.SET: return "set";
default: assert(false); return "";
}
}