UNPKG

@flowscripter/mpeg-sdl-parser

Version:

ISO/IEC 14496-34 Syntactic Description Language (MPEG SDL) parser implemented in TypeScript

130 lines (124 loc) 4.09 kB
import { Text } from "@codemirror/state"; import type { TreeCursor } from "@lezer/common"; import { InternalParseError } from "../../ParseError"; import { getChildNodesAndTokens, isAbstractNode, } from "../../util/nodeFactoryUtils"; import { NodeKind } from "../node/enum/node_kind"; import type { Token } from "../token/Token"; import type { AbstractStatement } from "../node/AbstractStatement"; import type { AbstractExpression } from "../node/AbstractExpression"; import type { NumberLiteral } from "../node/NumberLiteral"; import type { Identifier } from "../node/Identifier"; import { DoStatement } from "../node/DoStatement"; import { CompoundStatement } from "../node/CompoundStatement"; import { StatementKind } from "../node/enum/statement_kind"; export function getDoStatement( cursor: TreeCursor, text: Text, ): DoStatement { const childNodesAndTokens = getChildNodesAndTokens(cursor, text); let condition: AbstractExpression | NumberLiteral | Identifier | undefined; let compoundStatement: CompoundStatement | undefined; let doKeyword: Token | undefined; let openParenthesisPunctuator: Token | undefined; let closeParenthesisPunctuator: Token | undefined; let whileKeyword: Token | undefined; let semicolonPunctuator: Token | undefined; for (const childNodeOrToken of childNodesAndTokens) { if (isAbstractNode(childNodeOrToken)) { switch (childNodeOrToken.nodeKind) { case NodeKind.IDENTIFIER: condition = childNodeOrToken as Identifier; break; case NodeKind.NUMBER_LITERAL: condition = childNodeOrToken as NumberLiteral; break; case NodeKind.EXPRESSION: condition = childNodeOrToken as AbstractExpression; break; case NodeKind.STATEMENT: if ( (childNodeOrToken as AbstractStatement).statementKind === StatementKind.COMPOUND ) { compoundStatement = childNodeOrToken as CompoundStatement; } else { throw new InternalParseError( `Unexpected statement kind: ${ (childNodeOrToken as AbstractStatement).statementKind }`, ); } break; default: throw new InternalParseError( `Unexpected node kind: ${NodeKind[childNodeOrToken.nodeKind]}`, ); } } else { switch (childNodeOrToken.text) { case "do": doKeyword = childNodeOrToken; break; case "(": openParenthesisPunctuator = childNodeOrToken; break; case ")": closeParenthesisPunctuator = childNodeOrToken; break; case "while": whileKeyword = childNodeOrToken; break; case ";": semicolonPunctuator = childNodeOrToken; break; default: throw new InternalParseError( `Unexpected token: ${childNodeOrToken.text}`, ); } } } if (!condition) { throw new InternalParseError("Expected argument condition to be defined"); } if (!compoundStatement) { throw new InternalParseError( "Expected argument compoundStatement to be defined", ); } if (!doKeyword) { throw new InternalParseError("Expected argument doKeyword to be defined"); } if (!openParenthesisPunctuator) { throw new InternalParseError( "Expected argument openParenthesisPunctuator to be defined", ); } if (!closeParenthesisPunctuator) { throw new InternalParseError( "Expected argument closeParenthesisPunctuator to be defined", ); } if (!whileKeyword) { throw new InternalParseError( "Expected argument whileKeyword to be defined", ); } if (!semicolonPunctuator) { throw new InternalParseError( "Expected argument semicolonPunctuator to be defined", ); } return new DoStatement( compoundStatement, condition, doKeyword, whileKeyword, openParenthesisPunctuator, closeParenthesisPunctuator, semicolonPunctuator, ); }