123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163 |
- { *********************************************************************
- This file is part of the Free Component Library (FCL)
- Copyright (c) 2016 Michael Van Canneyt.
-
- Javascript parser
-
- See the file COPYING.FPC, included in this distribution,
- for details about the copyright.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
-
- **********************************************************************}
- unit jsparser;
- { $define debugparser}
- {$mode objfpc}{$H+}
- interface
- uses
- Classes, SysUtils, jsscanner, jstree, jstoken;
- Const
- SEmptyLabel = '';
- MinAsyncVersion = ecma2021;
- Type
- TECMAVersion = jsScanner.TECMAVersion;
- { TJSParser }
- TJSParser = Class(TObject)
- Private
- FFunctionDepth: Integer;
- FInput : TStream;
- FIsLHS: Boolean;
- FNoIn: Boolean;
- FScanner : TJSScanner;
- FPrevious,
- FCurrent : TJSToken;
- FCurrentString : String;
- FFreeScanner : Boolean;
- FCurrentVars : TJSElementNodes;
- FPeekToken: TJSToken;
- FPeekTokenString: String;
- FLabelSets,
- FCurrentLabelSet:TJSLabelSet;
- FLabels : TJSLabel;
- function CheckSemiColonInsert(aToken: TJSToken; Consume: Boolean): Boolean;
- function EnterLabel(ALabelName: String): TJSLabel;
- procedure Expect(aToken: TJSToken);
- procedure Consume(aToken: TJSToken; AllowSemicolonInsert : Boolean = False);
- procedure FreeCurrentLabelSet;
- function GetVersion: TECMAVersion;
- procedure LeaveLabel;
- function LookupLabel(ALabelName: String; Kind: TJSToken): TJSLabel;
- function ParseAdditiveExpression: TJSElement;
- function ParseArguments: TJSarguments;
- function ParseArrayLiteral: TJSElement;
- function ParseAssignmentExpression: TJSElement;
- function ParseBitwiseAndExpression: TJSElement;
- function ParseBitwiseORExpression: TJSElement;
- function ParseBitwiseXORExpression: TJSElement;
- function ParseBlock: TJSElement;
- function ParseBreakStatement: TJSElement;
- function ParseConditionalExpression: TJSElement;
- function ParseContinueStatement: TJSElement;
- function ParseEmptyStatement: TJSElement;
- function ParseEqualityExpression: TJSElement;
- function ParseExpression: TJSElement;
- function ParseExpressionStatement: TJSElement;
- function ParseFormalParameterList: TStrings;
- function ParseFunctionDeclaration: TJSFunctionDeclarationStatement;
- function ParseFunctionExpression: TJSFunctionDeclarationStatement;
- function ParseFunctionStatement: TJSElement;
- function ParseFunctionBody: TJSFunctionBody;
- function ParseIdentifier: String;
- function ParseIfStatement: TJSElement;
- function ParseIterationStatement: TJSElement;
- function ParseLabeledStatement: TJSElement;
- function ParseLeftHandSideExpression: TJSElement;
- function ParseLiteral: TJSElement;
- function ParseLogicalAndExpression: TJSElement;
- function ParseLogicalORExpression: TJSElement;
- function ParseMemberExpression: TJSElement;
- function ParseMultiplicativeExpression: TJSElement;
- function ParseNumericLiteral: TJSElement;
- function ParseObjectLiteral: TJSElement;
- function ParsePostFixExpression: TJSElement;
- function ParsePrimaryExpression: TJSElement;
- function ParseRegularExpressionLiteral: TJSElement;
- function ParseRelationalExpression: TJSElement;
- function ParseReturnStatement: TJSElement;
- function ParseShiftExpression: TJSElement;
- function ParseStatement: TJSElement;
- function ParseStatementList: TJSElement;
- function ParseStringLiteral: TJSElement;
- function ParseSwitchStatement: TJSElement;
- function ParseThrowStatement: TJSElement;
- function ParseTryStatement: TJSElement;
- function ParseUnaryExpression: TJSElement;
- function ParseVariableDeclaration: TJSElement;
- function ParseVariableDeclarationList: TJSElement;
- function ParseVariableStatement: TJSElement;
- function ParseWithStatement: TJSElement;
- Protected
- Procedure CheckParser;
- Function CurrentLabelSet : TJSLabelSet;
- function CurSource: String;
- Function CurLine : Integer;
- Function CurPos : Integer;
- Function CreateElement(AElementClass : TJSElementClass) : TJSElement;
- Procedure Error(Msg : String);
- Procedure Error(Fmt : String; Args : Array of const);
- // Parse functions
- function ParseSourceElements: TJSSourceElements;
- Property FunctionDepth : Integer Read FFunctionDepth Write FFunctionDepth;
- Property NoIn : Boolean Read FNoIn Write FNoIn;
- Property IsLHS : Boolean Read FIsLHS Write FIsLHS;
- Public
- Constructor Create(AInput: TStream; aVersion : TECMAVersion = ecma5);
- // Scanner has version
- Constructor Create(AScanner : TJSScanner);
- Destructor Destroy; override;
- Function Parse : TJSElement;
- Function ParseProgram : TJSFunctionDeclarationStatement;
- Function CurrentToken : TJSToken;
- Function CurrentTokenString : String;
- Function GetNextToken : TJSToken;
- Function PeekNextToken : TJSToken;
- Function IsEndOfLine : Boolean;
- Property ECMAVersion : TECMAVersion Read GetVersion;
- end;
- implementation
- uses typinfo;
- Resourcestring
- SErrUnmatchedCurlyBrace = 'Unmatched }';
- SErrUnmatchedSquareBrace = 'Unmatched ]';
- SErrUnmatchedBrace = 'Unmatched )';
- SErrUnexpectedToken = 'Unexpected token: ''%s''';
- SErrTokenMismatch = 'Unexpected token: ''%s'', expected: ''%s''';
- SErrSemicolonOrInExpected = 'Unexpected token: ''%s'', expected ; or ''in''';
- SErrSemicolonExpected = 'Unexpected token: ''%s'', expected ;';
- SErrDuplicateLabelName = 'Duplicate label name: ''%s''';
- SErrLabelNotContinuable = 'Label ''%s'' is not suitable for continue.';
- SErrLabelNOtDefinedOrReachable = 'Label ''%s'' is not defined or not reachable.';
- SErrContinueNotInLoop = 'Continue statement not in loop';
- SErrBreakNotInLoop = 'Break statement not in loop';
- SErrReturnNotInFunction = 'return statement not in a function body';
- SErrCaseEndExpected = 'Unexpected token: Expected }, case or default clause';
- SErrDuplicateSwitchDefault = 'Duplicate default clause for switch statement';
- SErrNewlineAfterThrow = 'Newline after throw not allowed';
- SErrCatchFinallyExpected = 'Unexpected token: Expected ''catch'' or ''finally''';
- SErrArgumentsExpected = 'Unexpected token: Expected '','' or '')'', got %s';
- SErrArrayEnd = 'Unexpected token: Expected '','' or '']'', got %s';
- //SErrObjectEnd = 'Unexpected token: Expected '','' or ''}'', got %s';
- SErrObjectElement = 'Unexpected token: Expected string, identifier or number after '','' got: %s';
- SErrLiteralExpected = 'Unexpected token: Expected: null, true, false, number, string, or regex, got: %s';
- SErrInvalidnumber = 'Invalid numerical value: %s';
- SErrInvalidRegularExpression = 'Invalid regular expression: %s';
- SErrFunctionNotAllowedHere = 'function keyword not allowed here';
- { TJSScanner }
- Function TJSParser.CurrentToken: TJSToken;
- begin
- Result:=FCurrent;
- end;
- Function TJSParser.CurrentTokenString: String;
- begin
- Result:=FCurrentString;
- end;
- Function TJSParser.GetNextToken: TJSToken;
- begin
- FPrevious:=FCurrent;
- If (FPeekToken<>tjsunknown) then
- begin
- FCurrent:=FPeekToken;
- FCurrentString:=FPeekTokenString;
- FPeekToken:=tjsUnknown;
- FPeekTokenString:='';
- end
- else
- begin
- FCurrent:=FScanner.FetchToken;
- FCurrentString:=FScanner.CurTokenString;
- end;
- Result:=FCurrent;
- {$ifdef debugparser}Writeln('GetNextToken (',FScanner.CurLine,',',FScanner.CurColumn,'): ',GetEnumName(TypeInfo(TJSToken),Ord(FCurrent)), ' As string: ',FCurrentString);{$endif debugparser}
- end;
- Function TJSParser.PeekNextToken: TJSToken;
- begin
- If (FPeekToken=tjsUnknown) then
- begin
- FPeekToken:=FScanner.FetchToken;
- FPeekTokenString:=FScanner.CurTokenString;
- end;
- {$ifdef debugparser}Writeln('PeekNextToken : ',GetEnumName(TypeInfo(TJSToken),Ord(FPeekToken)), ' As string: ',FPeekTokenString);{$endif debugparser}
- Result:=FPeekToken;
- end;
- Function TJSParser.IsEndOfLine: Boolean;
- begin
- Result:=FScanner.IsEndOfLine;
- end;
- Function TJSParser.CurPos: Integer;
- begin
- If Assigned(FScanner) then
- Result:=FScanner.CurColumn
- else
- Result:=0;
- end;
- Function TJSParser.CurLine: Integer;
- begin
- If Assigned(FScanner) then
- Result:=FScanner.CurRow
- else
- Result:=0;
- end;
- function TJSParser.CurSource: String;
- begin
- If Assigned(FScanner) then
- Result:=FScanner.CurFileName
- else
- Result:='';
- end;
- Procedure TJSParser.CheckParser;
- begin
- end;
- procedure TJSParser.LeaveLabel;
- Var
- L : TJSLabel;
- begin
- L:=FLabels;
- FLabels:=FLabels.Next;
- L.Free; // ??
- end;
- function TJSParser.LookupLabel(ALabelName : String; Kind : TJSToken) :TJSLabel;
- Var
- L : TJSLabel;
- begin
- L:=FLabels;
- Result:=Nil;
- While (L<>Nil) and (Result=Nil) do
- begin
- If (L.Name=ALabelName) then
- begin
- if (kind=tjsContinue) and (Not L.LabelSet.Continuable) and (ALabelName<>SEmptyLabel) then
- Error(SErrLabelNotContinuable,[ALabelName]);
- Result:=L;
- end;
- L:=L.Next;
- end;
- If (Result=Nil) then
- begin
- If (ALabelName<>'') then
- Error(SErrLabelNOtDefinedOrReachable,[ALabelName])
- else if kind=tjsCOntinue then
- Error(SErrContinueNotInLoop)
- else
- Error(SErrBreakNotInLoop);
- end;
- end;
- function TJSParser.EnterLabel(ALabelName : String) :TJSLabel;
- Var
- L : TJSLabel;
- begin
- If (ALAbelName<>SEmptyLabel) then
- begin
- L:=FLabels;
- While (L<>Nil) do
- begin
- If (L.Name=ALabelName) then
- Error(SErrDuplicateLabelName,[ALabelName]);
- L:=L.Next;
- end;
- end;
- L:=TJSLabel.Create;
- L.Name:=ALabelName;
- L.LabelSet:=CurrentLabelSet;
- L.LocationSource:=Self.CurSource;
- L.LocationLine:=CurLine;
- L.LocationPos:=CurPos;
- L.Next:=FLabels;
- FLabels:=L;
- Result:=L;
- end;
- Function TJSParser.CurrentLabelSet: TJSLabelSet;
- Var
- LS : TJSLabelSet;
- begin
- If (FCurrentLabelSet=Nil) then
- begin
- LS:=TJSLabelSet.Create;
- If (FLabelSets=Nil) then
- LS.Target:=1
- else
- LS.Target:=FLabelSets.Target;
- LS.Next:=FLabelSets;
- FLabelSets:=LS;
- FCurrentLabelSet:=LS;
- end;
- Result:=FCurrentLabelSet;
- end;
- Function TJSParser.CreateElement(AElementClass: TJSElementClass): TJSElement;
- begin
- Result:=AElementClass.Create(CurLine,CurPos,CurSource);
- end;
- Procedure TJSParser.Error(Msg: String);
- Var
- ErrAt : String;
- begin
- If Assigned(FScanner) then
- If FScanner.CurFilename<>'' then
- ErrAt:=Format('Error: file "%s" line %d, pos %d: ',[FScanner.CurFileName,FScanner.CurRow,FScanner.CurColumn])
- else
- ErrAt:=Format('Error: line %d, pos %d: ',[FScanner.Currow,FScanner.CurColumn]);
- Raise Exception.Create(ErrAt+Msg)
- end;
- Procedure TJSParser.Error(Fmt: String; Args: Array of const);
- begin
- Error(Format(Fmt,Args));
- end;
- Constructor TJSParser.Create(AInput: TStream; aVersion : TECMAVersion = ecma5);
- begin
- FInput:=AInput;
- FCurrent:=TJSUnknown;
- FScanner:=TJSScanner.Create(FInput,aVersion);
- FFreeScanner:=True;
- end;
- Constructor TJSParser.Create(AScanner: TJSScanner);
- begin
- FCurrent:=TJSUnknown;
- FScanner:=AScanner;
- FFreeScanner:=False;
- end;
- Destructor TJSParser.Destroy;
- begin
- if FFreeScanner then
- FreeAndNil(FScanner);
- inherited;
- end;
- procedure TJSParser.Expect(aToken: TJSToken);
- begin
- {$ifdef debugparser} Writeln('Expecting : ',GetEnumName(TypeInfo(TJSToken),Ord(AToken)), ' As string: ',TokenInfos[AToken]);{$endif debugparser}
- If Not CheckSemiColonInsert(AToken,False) then
- if (CurrentToken<>aToken) then
- Error(SerrTokenMismatch,[CurrenttokenString,TokenInfos[aToken]]);
- end;
- function TJSParser.CheckSemiColonInsert(aToken : TJSToken; Consume : Boolean) : Boolean;
- begin
- Result:=(AToken=tjsSemiColon);
- If Result then
- begin
- Result:=(CurrentToken=tjsCurlyBraceClose) or (FScanner.WasEndOfLine) or (CurrentToken=tjsEOF);
- If Result and Consume then
- FPrevious:=tjsSemiColon;
- end;
- end;
- procedure TJSParser.Consume(aToken: TJSToken; AllowSemicolonInsert: Boolean);
- begin
- {$ifdef debugparser} Writeln('Consuming : ',GetEnumName(TypeInfo(TJSToken),Ord(AToken)), ' As string: ',TokenInfos[AToken]);{$endif debugparser}
- Expect(aToken);
- If not (AllowSemiColonInsert and CheckSemiColonInsert(aToken,True)) then
- GetNextToken;
- end;
- function TJSParser.ParseIdentifier : String;
- begin
- Result:='';
- Repeat
- Expect(tjsIdentifier);
- Result:=Result+CurrentTokenString;
- GetNextToken;
- If (CurrentToken=tjsDot) then
- begin
- If (Result<>'') then
- Result:=Result+'.';
- GetNextToken;
- end;
- until (CurrentToken<>tjsIdentifier);
- end;
- function TJSParser.ParseFormalParameterList : TStrings;
- begin
- Result:=Nil;
- While (CurrentToken=tjsIdentifier) do
- begin
- Expect(tjsIdentifier);
- If (Result=Nil) then
- Result:=TstringList.Create;
- Result.Add(CurrentTokenString);
- GetNextToken;
- If (CurrentToken=tjsComma) then
- GetNextToken;
- end;
- end;
- function TJSParser.ParseFunctionDeclaration : TJSFunctionDeclarationStatement;
- Var
- Id : String;
- D : TJSFuncDef;
- args : TStrings;
- body : TJSFunctionBody;
- begin
- {$ifdef debugparser} Writeln('>>> Entering ParseFunctionDeclaration');{$endif debugparser}
- Consume(tjsFunction);
- ID:=ParseIdentifier;
- Consume(tjsBraceOpen);
- Args:=ParseFormalParameterList;
- try
- Consume(tjsBraceClose);
- Consume(tjsCurlyBraceOpen);
- Inc(FFunctionDepth);
- try
- Body:=ParseFunctionBody;
- try
- // GetNextToken; not sure
- Consume(tjsCurlyBraceClose);
- D:=TJSFuncDef.Create;
- try
- D.Name:=ID;
- If Assigned(Args)then
- D.Params.Assign(Args);
- D.Body:=Body;
- Result:=TJSFunctionDeclarationStatement(CreateElement(TJSFunctionDeclarationStatement));
- Result.AFunction:=D;
- except
- FreeAndNil(D);
- Raise;
- end;
- except
- FreeAndNil(Body);
- Raise;
- end;
- finally
- Dec(FFunctionDepth);
- end;
- finally
- FreeAndNil(Args);
- end;
- {$ifdef debugparser} Writeln('>>> Exiting ParseFunctionDeclaration');{$endif debugparser}
- end;
- function TJSParser.ParseStatementList : TJSElement;
- Var
- E : TJSElement;
- SL : TJSSTatementList;
- begin
- {$ifdef debugparser} Writeln('>>> ParseStatementList');{$endif debugparser}
- E:=ParseStatement;
- try
- if (CurrentToken in [tjsCurlyBraceClose,tjsEof,tjsCase,tjsDefault]) then
- Result:=E
- else
- begin
- SL:=TJSSTatementList(CreateElement(TJSStatementList));
- try
- SL.A:=E;
- SL.B:=ParseStatementlist();
- Result:=SL;
- except
- FreeAndNil(SL);
- Raise;
- end;
- end;
- except
- FreeAndNil(E);
- Raise;
- end;
- {$ifdef debugparser} Writeln('<<< ParseStatementList');{$endif debugparser}
- end;
- function TJSParser.ParseBlock : TJSElement;
- begin
- {$ifdef debugparser} Writeln('>>> ParseBlock');{$endif debugparser}
- Consume(tjsCurlyBraceOpen);
- If (CurrentToken=tjsCurlyBraceClose) then
- Result:=CreateElement(TJSEmptyBlockStatement)
- else
- result:=ParseStatementList;
- Consume(tjsCurlyBraceClose);
- {$ifdef debugparser} Writeln('<<< ParseBlock');{$endif debugparser}
- end;
- function TJSParser.ParseArrayLiteral: TJSElement;
- Var
- N : TJSArrayLiteral;
- E : TJSArrayLiteralElement;
- I : Integer;
- begin
- Consume(tjsSquaredBraceOpen);
- N:=TJSArrayLiteral(CreateElement(TJSArrayLiteral));
- Result:=N;
- try
- I:=0;
- While (CurrentToken<>tjsSquaredBraceClose) do
- begin
- If (CurrentToken=tjsComma) then
- begin
- GetNextToken;
- Inc(I);
- end
- else
- begin
- E:=N.Elements.AddElement;
- E.ElementIndex:=I;
- Inc(I);
- E.Expr:=ParseAssignmentExpression;
- If Not (CurrentToken in [tjsComma,tjsSquaredBraceClose]) then
- Error(SErrArrayEnd,[CurrentTokenString])
- end;
- end;
- Consume(tjsSquaredBraceClose);
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseObjectLiteral: TJSElement;
- Var
- N : TJSObjectLiteral;
- E : TJSObjectLiteralElement;
- begin
- Consume(tjsCurlyBraceOpen);
- N:=TJSObjectLiteral(CreateElement(TJSObjectLiteral));
- Result:=N;
- try
- While (CurrentToken<>tjsCurlyBraceClose) do
- begin
- While CurrentToken=tjsComma do
- GetNextToken;
- If (CurrentToken in [tjsIdentifier,jstoken.tjsString,tjsnumber]) then
- begin
- E:=N.Elements.AddElement;
- E.Name:=CurrenttokenString;
- GetNextToken;
- end
- else
- Error(SErrObjectElement,[CurrentTokenString]);
- Consume(tjsColon);
- E.Expr:=ParseAssignmentExpression;
- While CurrentToken=tjsComma do
- GetNextToken;
- { If Not (CurrentToken in [tjsComma,tjsCurlyBraceClose]) then
- Error(SErrObjectEnd,[CurrentTokenString])}
- end;
- Consume(tjsCurlyBraceClose);
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseNumericLiteral: TJSElement;
- Var
- L : TJSLiteral;
- D : Double;
- I : Integer;
- begin
- {$ifdef debugparser} Writeln('Parsing numerical literal');{$endif debugparser}
- Result:=Nil;
- try
- Val(CurrentTokenString,D,I);
- If (I>0) then
- Error(SErrInvalidnumber,[CurrentTokenString]);
- L:=TJSLiteral(CreateElement(TJSLiteral));
- GetNextToken;
- L.Value.AsNumber:=D;
- Result:=L;
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseStringLiteral: TJSElement;
- Var
- L : TJSLiteral;
- begin
- {$ifdef debugparser} Writeln('Parsing string literal');{$endif debugparser}
- Result:=Nil;
- try
- L:=TJSLiteral(CreateElement(TJSLiteral));
- L.Value.AsString:=CurrentTokenString;
- GetNextToken;
- Result:=L;
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseRegularExpressionLiteral: TJSElement;
- Var
- S,pa,fl : String;
- P : integer;
- R : TJSRegularExpressionLiteral;
- begin
- Result:=Nil;
- If (CurrentToken=tjsRegex) then
- begin
- S:=CurrentTokenString;
- P:=Length(S);
- While (P>=1) and (S[P]<>'/') do
- Dec(P);
- If (P<=1) then
- Error(SErrInvalidRegularExpression,[CurrentTokenString]);
- pa:=Copy(S,2,P-1);
- fl:=Copy(S,P,Length(S)-P+1);
- R:=TJSRegularExpressionLiteral(CreateElement(TJSRegularExpressionLiteral));
- Result:=R;
- R.Pattern.AsString:=Pa;
- R.PatternFlags.AsString:=Fl;
- R.Argv[0]:=R.Pattern;
- R.Argv[1]:=R.PatternFlags;
- end;
- try
- Consume(tjsRegEx);
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseLiteral: TJSElement;
- Var
- L : TJSLiteral;
- begin
- {$ifdef debugparser}Writeln('Parsing literal: ',GetEnumName(TypeInfo(TJSToken),Ord(CurrentToken)), ' As string: ',CurrentTokenString);{$endif debugparser}
- Result:=Nil;
- Case CurrentToken of
- tjsNull : begin
- L:=TJSLiteral(CreateElement(TJSLiteral));
- Result:=L;
- L.Value.IsNull:=True;
- GetNextToken;
- end;
- tjsTrue,
- tjsFalse: begin
- L:=TJSLiteral(CreateElement(TJSLiteral));
- Result:=L;
- L.Value.AsBoolean:=(CurrentToken=tjsTrue);
- GetNextToken;
- end;
- tjsNumber : Result:=ParseNumericLiteral;
- jstoken.tjsString : Result:=ParseStringLiteral;
- tjsDiv,
- tjsDivEq : Result:=ParseRegularExpressionLiteral
- else
- Error(SErrLiteralExpected,[CurrentTokenString]);
- end;
- end;
- function TJSParser.ParsePrimaryExpression: TJSElement;
- Var
- R : TJSPrimaryExpressionIdent;
- begin
- {$ifdef debugparser} Writeln('ParsePrimaryExpression');{$endif debugparser}
- Result:=Nil;
- try
- Case CurrentToken of
- tjsThis :
- begin
- Result:=TJSPrimaryExpressionThis(CreateElement(TJSPrimaryExpressionThis));
- GetNextToken;
- end;
- tjsidentifier:
- begin
- R:=TJSPrimaryExpressionIdent(CreateElement(TJSPrimaryExpressionIdent));
- Result:=R;
- R.Name:=CurrentTokenString;
- GetNextToken;
- end;
- tjsSquaredBraceOpen: Result:=ParseArrayLiteral;
- tjsCurlyBraceOpen: Result:=ParseObjectLiteral;
- tjsBraceOpen:
- begin
- Consume(tjsBraceOpen);
- Result:=ParseExpression;
- Consume(tjsBraceClose);
- end;
- else
- Result:=ParseLiteral;
- end; // Case;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParsePrimaryExpression');{$endif debugparser}
- end;
- function TJSParser.ParseMemberExpression: TJSElement;
- Var
- M : TJSDotMemberExpression;
- N : TJSNewMemberExpression;
- B : TJSBracketMemberExpression;
- Done : Boolean;
- begin
- {$ifdef debugparser} Writeln('ParseMemberExpression');{$endif debugparser}
- Case CurrentToken of
- tjsFunction : Result:=ParseFunctionExpression();
- tjsNew : begin
- GetNextToken;
- N:=TJSNewMemberExpression(CreateElement(TJSNewMemberExpression));
- try
- Result:=N;
- N.MExpr:=ParseMemberExpression();
- if (CurrentToken=tjsBraceOpen) then
- N.Args:=ParseArguments;
- except
- FreeAndNil(N);
- Raise;
- end;
- end;
- else
- Result:=ParsePrimaryExpression()
- end;
- try
- Done:=False;
- Repeat
- Case CurrentToken of
- tjsDot :
- begin
- M:=TJSDotMemberExpression(CreateElement(TJSDotMemberExpression));
- M.MExpr:=Result;
- Result:=M;
- GetNextToken;
- If (CurrentToken=tjsIdentifier) then
- M.Name:=CurrentTokenString;
- Consume(tjsIdentifier);
- end;
- tjsSquaredBraceOpen:
- begin
- B:=TJSBracketMemberExpression(CreateElement(TJSBracketMemberExpression));
- B.MExpr:=Result;
- Result:=B;
- GetNextToken;
- B.Name:=ParseExpression();
- Consume(tjsSquaredBraceClose);
- end;
- else
- Done:=True;
- isLHS:=True;
- end;
- Until Done;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseMemberExpression');{$endif debugparser}
- end;
- function TJSParser.ParseArguments: TJSarguments;
- Var
- E : TJSArrayLiteralElement;
- begin
- Consume(tjsBraceOpen);
- Result:=TJSArguments(CreateElement(TJSArguments));
- try
- While (CurrentToken<>tjsBraceClose) do
- begin
- E:=Result.Elements.AddElement;
- E.Expr:=ParseAssignmentExpression;
- If (CurrentToken<>tjsBraceClose) then
- If CurrentToken=tjsComma then
- GetNextToken
- else
- Error(SErrArgumentsExpected,[CurrentTokenString]);
- end;
- Consume(tjsBraceClose);
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseLeftHandSideExpression: TJSElement;
- Var
- M : TJSDotMemberExpression;
- B : TJSBracketMemberExpression;
- C : TJSCallExpression;
- Done : Boolean;
- begin
- {$ifdef debugparser} Writeln('ParseLeftHandSideExpression');{$endif debugparser}
- Case CurrentToken of
- tjsFunction : Result:=ParseFunctionExpression;
- tjsNew : Result:=ParseMemberExpression;
- else
- Result:=ParsePrimaryExpression
- end;
- try
- Done:=False;
- Repeat
- Case CurrentToken of
- tjsDot :
- begin
- M:=TJSDotMemberExpression(CreateElement(TJSDotMemberExpression));
- M.MExpr:=Result;
- Result:=M;
- GetNextToken;
- If (CurrentToken=tjsIdentifier) then
- M.Name:=CurrentTokenString;
- Consume(tjsIdentifier);
- end;
- tjsSquaredBraceOpen:
- begin
- B:=TJSBracketMemberExpression(CreateElement(TJSBracketMemberExpression));
- B.MExpr:=Result;
- Result:=B;
- GetNextToken;
- B.Name:=ParseExpression;
- Consume(tjsSquaredBraceClose);
- end;
- tjsBraceOpen:
- begin
- C:=TJSCallExpression(CreateElement(TJSCallExpression));
- C.Expr:=Result;
- Result:=C;
- C.Args:=ParseArguments;
- end;
- else
- {$ifdef debugparser}Writeln('Leaving LHS');{$endif debugparser}
- Done:=True;
- isLHS:=True;
- end;
- Until Done;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseLeftHandSideExpression');{$endif debugparser}
- end;
- function TJSParser.ParsePostFixExpression: TJSElement;
- Var
- R : TJSUnaryExpression;
- begin
- {$ifdef debugparser} Writeln('ParsePostfixExpression');{$endif debugparser}
- Result:=ParseLeftHandSideExpression;
- Try
- If (Not IsEndOfLine) and (CurrentToken in [tjsPlusPlus,tjsMinusMinus]) then
- begin
- If (CurrentToken=tjsPlusPLus) then
- R:=TJSUnaryExpression(CreateElement(TJSUnaryPostPlusPlusExpression))
- else
- R:=TJSUnaryExpression(CreateElement(TJSUnaryPostMinusMinusExpression));
- R.A:=Result;
- Result:=R;
- GetNextToken;
- isLHS:=False;
- end;
- except
- freeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParsePostfixExpression');{$endif debugparser}
- end;
- function TJSParser.ParseUnaryExpression: TJSElement;
- Var
- C : TJSElementClass;
- R : TJSUnaryExpression;
- begin
- {$ifdef debugparser} Writeln('ParseUnaryExpression');{$endif debugparser}
- C:=Nil;
- Result:=Nil;
- try
- Case CurrentToken of
- tjsDelete : C:=TJSUnaryDeleteExpression;
- tjsVoid : C:=TJSUnaryVoidExpression;
- tjsTypeOf : C:=TJSUnaryTypeOfExpression;
- tjsPlusPlus : C:=TJSUnaryPrePlusPlusExpression;
- tjsMinusMinus : C:=TJSUnaryPreMinusMinusExpression;
- tjsPlus : C:=TJSUnaryPlusExpression;
- tjsMinus : C:=TJSUnaryMinusExpression;
- tjsInv : C:=TJSUnaryInvExpression;
- tjsNot : C:=TJSUnaryNotExpression;
- else
- Result:=ParsePostFixExpression;
- end;
- If (Result=Nil) then
- begin
- {$ifdef debugparser} Writeln('Found Unary Expression',GetEnumName(TypeInfo(TJSToken),Ord(CurrentToken)), ' As string: ',CurrentTokenString);{$endif debugparser}
- R:=TJSUnaryExpression(CreateElement(C));
- Result:=R;
- GetNextToken;
- R.A:=ParseUnaryExpression();
- isLHS:=False;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseUnaryExpression');{$endif debugparser}
- end;
- function TJSParser.ParseMultiplicativeExpression: TJSElement;
- Var
- C : TJSElementClass;
- R : TJSMultiplicativeExpression;
- begin
- {$ifdef debugparser} Writeln('ParseMultiplicativeExpression');{$endif debugparser}
- Result:=ParseUnaryExpression;
- try
- While (CurrentToken in [tjsMul,tjsDiv,tjsMod]) do
- begin
- if CurrentToken=tjsMul then
- C:=TJSMultiplicativeExpressionMul
- else if CurrentToken=tjsDiv then
- C:=TJSMultiplicativeExpressionDiv
- else
- C:=TJSMultiplicativeExpressionMod;
- R:=TJSMultiplicativeExpression(CreateElement(C));
- GetNextToken;
- R.A:=Result;
- Result:=R;
- R.B:=ParseUnaryExpression;
- isLHS:=False;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseMultiplicativeExpression');{$endif debugparser}
- end;
- function TJSParser.ParseAdditiveExpression: TJSElement;
- Var
- C : TJSElementClass;
- R : TJSAdditiveExpression;
- begin
- {$ifdef debugparser} Writeln('ParseAdditiveExpression');{$endif debugparser}
- Result:=ParseMultiplicativeExpression;
- try
- While (CurrentToken in [tjsPlus,tjsMinus]) do
- begin
- if CurrentToken=tjsPlus then
- C:=TJSAdditiveExpressionPlus
- else
- C:=TJSAdditiveExpressionMinus;
- R:=TJSAdditiveExpression(CreateElement(C));
- GetNextToken;
- R.A:=Result;
- Result:=R;
- R.B:=ParseMultiplicativeExpression;
- isLHS:=False;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseAdditiveExpression');{$endif debugparser}
- end;
- function TJSParser.ParseShiftExpression: TJSElement;
- Var
- C : TJSElementClass;
- R : TJSShiftExpression;
- begin
- {$ifdef debugparser} Writeln('ParseShiftExpression');{$endif debugparser}
- Result:=ParseAdditiveExpression;
- try
- While (CurrentToken in [tjsLshift,tjsRshift,tjsURShift]) do
- begin
- Case CurrentToken of
- tjsLshift : C:=TJSLShiftExpression;
- tjsRshift : C:=TJSRShiftExpression;
- tjsURshift : C:=TJSURShiftExpression;
- end;
- R:=TJSShiftExpression(CreateElement(C));
- R.A:=Result;
- Result:=R;
- GetNextToken;
- R.B:=ParseAdditiveExpression;
- IsLHS:=False;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseShiftExpression');{$endif debugparser}
- end;
- function TJSParser.ParseRelationalExpression: TJSElement;
- Var
- S : Set of TJSToken;
- C : TJSElementClass;
- R : TJSRelationalExpression;
- begin
- {$ifdef debugparser} Writeln('ParseRelationalExpression');{$endif debugparser}
- Result:=ParseShiftExpression;
- try
- S:=[tjsLT,tjsGT,tjsLE,tjsGE,tjsInstanceOf];
- If Not Noin then
- Include(S,tjsIn);
- While (CurrentToken in S) do
- begin
- Case CurrentToken of
- tjsLT : C:=TJSRelationalExpressionLT;
- tjsGT : C:=TJSRelationalExpressionGT;
- tjsLE : C:=TJSRelationalExpressionLE;
- tjsGE : C:=TJSRelationalExpressionGE;
- tjsInstanceOf :C:=TJSRelationalExpressionInstanceOf;
- tjsIn : C:=TJSRelationalExpressionIn;
- end;
- R:=TJSRelationalExpression(CreateElement(C));
- R.A:=Result;
- Result:=R;
- GetNextToken;
- R.B:=ParseRelationalExpression();
- IsLHS:=False;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseRelationalExpression');{$endif debugparser}
- end;
- function TJSParser.ParseEqualityExpression: TJSElement;
- Var
- C : TJSElementClass;
- E : TJSEqualityExpression;
- begin
- {$ifdef debugparser} Writeln('ParseEqualityExpression');{$endif debugparser}
- Result:=ParseRelationalExpression;
- try
- While (CurrentToken in [tjsEq,tjsNE,tjsSEQ,tjsSNE]) do
- begin
- Case CurrentToken of
- tjsEq : C:=TJSEqualityExpressionEQ;
- tjsNE : C:=TJSEqualityExpressionNE;
- tjsSEQ : C:=TJSEqualityExpressionSEQ;
- tjsSNE : C:=TJSEqualityExpressionSNE;
- end;
- GetNextToken;
- E:=TJSEqualityExpression(CreateElement(C));
- Result:=E;
- E.A:=Result;
- E.B:=ParseEqualityExpression();
- E:=Nil;
- IsLHS:=False;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseEqualityExpression');{$endif debugparser}
- end;
- function TJSParser.ParseBitwiseAndExpression: TJSElement;
- Var
- L : TJSBitwiseAndExpression;
- begin
- {$ifdef debugparser} Writeln('ParseBitwiseAndExpression');{$endif debugparser}
- Result:=ParseEqualityExpression;
- try
- If (CurrentToken<>tjsAnd) then
- exit;
- GetNextToken;
- L:=TJSBitwiseAndExpression(CreateElement(TJSBitwiseAndExpression));
- L.A:=Result;
- Result:=L;
- L.B:=ParseBitwiseAndExpression();
- IsLHS:=False;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseBitwiseAndExpression');{$endif debugparser}
- end;
- function TJSParser.ParseBitwiseXORExpression: TJSElement;
- Var
- L : TJSBitwiseXOrExpression;
- begin
- {$ifdef debugparser} Writeln('ParseBitwiseXorExpression');{$endif debugparser}
- Result:=ParseBitwiseAndExpression;
- try
- If (CurrentToken<>tjsXOr) then
- exit;
- GetNextToken;
- L:=TJSBitwiseXOrExpression(CreateElement(TJSBitwiseXOrExpression));
- L.A:=Result;
- Result:=L;
- L.B:=ParseBitwiseXORExpression();
- IsLHS:=False;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseBitwiseXorExpression');{$endif debugparser}
- end;
- function TJSParser.ParseBitwiseORExpression: TJSElement;
- Var
- L : TJSBitwiseOrExpression;
- begin
- {$ifdef debugparser} Writeln('ParseBitWiseOrExpression');{$endif debugparser}
- Result:=ParseBitwiseXORExpression;
- try
- If (CurrentToken<>tjsOr) then
- exit;
- GetNextToken;
- L:=TJSBitwiseOrExpression(CreateElement(TJSBitwiseOrExpression));
- L.A:=Result;
- Result:=L;
- L.B:=ParseBitwiseORExpression();
- IsLHS:=False;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseBitWiseOrExpression');{$endif debugparser}
- end;
- function TJSParser.ParseLogicalAndExpression: TJSElement;
- Var
- L : TJSLogicalAndExpression;
- begin
- {$ifdef debugparser} Writeln('ParseLogicalAndExpression');{$endif debugparser}
- Result:=ParseBitwiseORExpression;
- try
- If (CurrentToken<>tjsAndAnd) then
- exit;
- GetNextToken;
- L:=TJSLogicalAndExpression(CreateElement(TJSLogicalAndExpression));
- L.A:=Result;
- Result:=L;
- L.B:=ParseLogicalAndExpression();
- IsLHS:=False;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseLogicalAndExpression');{$endif debugparser}
- end;
- function TJSParser.ParseLogicalORExpression: TJSElement;
- Var
- L : TJSLogicalOrExpression;
- begin
- {$ifdef debugparser} Writeln('ParseLogicalOrExpression');{$endif debugparser}
- Result:=ParseLogicalAndExpression;
- try
- If (CurrentToken<>tjsOROR) then
- exit;
- GetNextToken;
- L:=TJSLogicalOrExpression(CreateElement(TJSLogicalOrExpression));
- L.A:=Result;
- Result:=L;
- L.B:=ParseLogicalOrExpression();
- IsLHS:=False;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseLogicalOrExpression');{$endif debugparser}
- end;
- function TJSParser.ParseConditionalExpression: TJSElement;
- Var
- N : TJSConditionalExpression;
- L : TJSElement;
- begin
- {$ifdef debugparser} Writeln('ParseConditionalExpression');{$endif debugparser}
- Result:=Nil;
- Result:=ParseLogicalORExpression;
- try
- If (CurrentToken=tjsConditional) then
- begin
- {$ifdef debugparser} Writeln('ParseConditionalExpression : Detected conditional ');{$endif debugparser}
- GetNextToken;
- L:=Result;
- N:=TJSConditionalExpression(CreateElement(TJSConditionalExpression));
- Result:=N;
- N.A:=L;
- L:=Nil;
- N.B:=ParseAssignmentExpression;
- Consume(tjsColon);
- N.C:=ParseAssignmentExpression;
- IsLHS:=False;
- end;
- except
- FreeandNil(Result);
- end;
- {$ifdef debugparser} Writeln('Exit ParseConditionalExpression');{$endif debugparser}
- end;
- function TJSParser.ParseAssignmentExpression: TJSElement;
- Var
- N : TJSElement;
- C : TJSElementClass;
- A : TJSAssignStatement;
- begin
- {$ifdef debugparser} Writeln('ParseAssignmentExpression');{$endif debugparser}
- Result:=Nil;
- N:=ParseConditionalExpression;
- If not isLHS then
- Result:=N
- else
- Case CurrentToken of
- tjsAssign : C:=TJSSimpleAssignStatement;
- tjsMulEq : C:=TJSMulEqAssignStatement;
- tjsDivEq : C:=TJSDivEqAssignStatement;
- tjsModEq : C:=TJSModEqAssignStatement;
- tjsPlusEq : C:=TJSAddEqAssignStatement;
- tjsMinusEq : C:=TJSSubEqAssignStatement;
- tjsLShiftEq : C:=TJSLShiftEqAssignStatement;
- tjsRShiftEq : C:=TJSRShiftEqAssignStatement;
- tjsURShiftEq : C:=TJSURShiftEqAssignStatement;
- tjsANDEq : C:=TJSANDEqAssignStatement;
- tjsOREq : C:=TJSOREqAssignStatement;
- tjsXOREq : C:=TJSXOREqAssignStatement;
- else
- // writeln('Strange token',GetEnumName(TypeInfo(TJSToken),Ord(CurrentToken)), ' As string: ',CurrentTokenString);
- Result:=N
- end;
- If Result<>Nil then
- begin
- {$ifdef debugparser} Writeln('Exit ParseAssignmentExpression - no assignment');{$endif debugparser}
- Exit;
- end;
- A:=TJSAssignStatement(CreateElement(C));
- try
- Result:=A;
- A.Lhs:=N;
- GetNextToken;
- {$ifdef debugparser} Writeln('ParseAssignmentExpression - level 2');{$endif debugparser}
- N:=ParseAssignmentExpression();
- {$ifdef debugparser} Writeln('Exit ParseAssignmentExpression - level 2');{$endif debugparser}
- A.Expr:=N;
- IsLhs:=False;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseAssignmentExpression');{$endif debugparser}
- end;
- function TJSParser.ParseVariableDeclaration: TJSElement;
- Var
- V : TJSVarDeclaration;
- begin
- {$ifdef debugparser} Writeln('ParseVariableDeclaration');{$endif debugparser}
- V:=TJSVarDeclaration(CreateElement(TJSVarDeclaration));;
- try
- V.Name:=CurrenttokenString;
- Consume(tjsIdentifier);
- if (CurrentToken=tjsAssign) then
- begin
- GetNextToken;
- V.Init:=ParseAssignmentExpression;
- end;
- Result:=V;
- FCurrentVars.AddNode.Node:=Result;
- except
- FreeAndNil(V);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseVariableDeclaration');{$endif debugparser}
- end;
- function TJSParser.ParseVariableDeclarationList: TJSElement;
- Var
- E,N : TJSElement;
- L : TJSVariableDeclarationList;
- begin
- {$ifdef debugparser} Writeln('ParseVariableDeclarationList entry');{$endif debugparser}
- E:=ParseVariableDeclaration;
- If (CurrentToken<>tjsComma) then
- Result:=E
- else
- begin
- L:=TJSVariableDeclarationList(CreateElement(TJSVariableDeclarationList));
- Result:=L;
- try
- Consume(tjsComma);
- N:=ParseVariableDeclarationList();
- L.A:=E;
- L.B:=N;
- except
- FreeAndNil(Result);
- Raise;
- end;
- end;
- {$ifdef debugparser} Writeln('ParseVariableDeclarationList exit');{$endif debugparser}
- end;
- function TJSParser.ParseVariableStatement : TJSElement;
- Var
- V : TJSVariableStatement;
- begin
- {$ifdef debugparser} Writeln('ParseVariableStatement entry');{$endif debugparser}
- Result:=Nil;
- Consume(tjsVar);
- Result:=ParseVariableDeclarationList;
- try
- Consume(tjsSemicolon,true);
- V:=TJSVariableStatement(CreateElement(TJSVariableStatement));
- V.A:=Result;
- Result:=V;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('ParseVariableStatement exit');{$endif debugparser}
- end;
- function TJSParser.ParseEmptyStatement : TJSElement;
- begin
- Consume(tjsSemiColon,true);
- Result:=CreateElement(TJSEmptyStatement);
- end;
- function TJSParser.ParseIfStatement : TJSElement;
- Var
- C,BTrue,BFalse : TJSElement;
- I : TJSIFstatement;
- begin
- C:=Nil;
- BTrue:=Nil;
- BFalse:=Nil;
- try
- Consume(tjsIF);
- Consume(tjsBraceOpen);
- C:=ParseExpression;
- Consume(tjsBraceClose);
- BTrue:=ParseStatement;
- If (CurrentToken=tjsElse) then
- begin
- Consume(tjsElse);
- BFalse:=ParseStatement;
- end;
- I:=TJSIfStatement(CreateElement(TJSIfStatement));
- I.Cond:=C;
- I.BTrue:=Btrue;
- I.BFalse:=BFalse;
- Result:=I;
- except
- FreeAndNil(C);
- FreeAndNil(BTrue);
- FreeAndNil(BFalse);
- Raise;
- end;
- end;
- function TJSParser.ParseIterationStatement : TJSElement;
- Var
- F : TJSForStatement;
- FI : TJSForInStatement;
- W : TJSWhileStatement;
- N : TJSElement;
- begin
- Result:=Nil;
- N:=Nil;
- CurrentLabelSet.Continuable:=True;
- EnterLabel(SEmptyLabel);
- try
- try
- Case CurrentToken of
- tjsDo :
- begin
- GetNextToken;
- W:=TJSDoWhileStatement(CreateElement(TJSDoWhileStatement));
- Result:=W;
- W.Body:=ParseStatement;
- Consume(tjsWhile);
- Consume(tjsBraceOpen);
- W.Cond:=ParseExpression;
- Consume(tjsBraceClose);
- Consume(tjsSemicolon,True);
- end;
- tjsWhile :
- begin
- GetNextToken;
- W:=TJSWhileStatement(CreateElement(TJSWhileStatement));
- Result:=W;
- Consume(tjsBraceOpen);
- W.Cond:=ParseExpression;
- Consume(tjsBraceClose);
- W.Body:=ParseStatement;
- Result:=W;
- end;
- else
- // For ?
- GetNextToken;
- Consume(tjsBraceopen);
- If (CurrentToken=tjsVar) then
- begin
- GetNextToken;
- N:=ParseVariableDeclarationList;
- // for (var in
- If (CurrentToken=tjsIn) and (N is tJSVarDeclaration) then
- begin
- Fi:=TJSForInStatement(CreateElement(TJSForInStatement));
- Result:=Fi;
- Fi.LHS:=N;
- GetNextToken;
- Fi.List:=ParseExpression;
- Consume(tjsBraceClose);
- Fi.Body:=ParseStatement;
- end;
- // for (var ;
- If (CurrentToken<>tjsSemicolon) then
- If (N is tJSVarDeclaration) then
- Error(SErrSemicolonOrInExpected,[CurrentTokenString])
- else
- Error(SErrSemicolonExpected,[CurrentTokenString]);
- GetNextToken;
- F:=TJSForStatement(CreateElement(TJSForStatement));
- Result:=F;
- If (CurrentToken<>tjsSemicolon) then
- F.Cond:=ParseExpression;
- Consume(tjsSemicolon);
- If (CurrentToken<>tjsBraceClose) then
- F.Incr:=ParseExpression;
- Consume(tjsBraceClose);
- F.Body:=ParseStatement;
- end
- else
- begin
- If (CurrentToken<>tjsSemicolon) then
- begin
- N:=ParseExpression;
- If (CurrentToken=tjsIn) then
- begin
- Fi:=TJSForInStatement(CreateElement(TJSForInStatement));
- Result:=Fi;
- Fi.LHS:=N;
- N:=Nil; // prevent freeing a second time in case of an exception.
- GetNextToken;
- Fi.List:=ParseExpression;
- Consume(tjsBraceClose);
- Fi.Body:=ParseStatement;
- Exit; // We must jump out here
- end
- end
- else
- N:=Nil;
- // For ( Init; Cond; incr)
- F:=TJSForStatement(CreateElement(TJSForStatement));
- Result:=F;
- F.Init:=N;
- N:=Nil; // prevent freeing a second time in case of an exception.
- Consume(tjsSemicolon);
- if (CurrentToken<>tjsSemicolon) then
- F.Cond:=ParseExpression;
- Consume(tjsSemicolon);
- If (CurrentToken<>tjsBraceClose) Then
- F.Incr:=ParseExpression;
- Consume(tjsBraceClose);
- F.Body:=ParseStatement;
- end;
- end; // Case
- Finally
- LeaveLabel;
- FreeCurrentLabelSet;
- end;
- except
- FreeAndNil(N);
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseContinueStatement : TJSElement;
- Var
- L : TJSLabel;
- C : TJSContinueStatement;
- begin
- C:=TJSContinueStatement(CreateElement(TJSContinueStatement));
- try
- Result:=C;
- Consume(tjsContinue);
- If (CurrentToken=tjsSemicolon) then
- L:=LookupLabel(SEmptyLabel,tjsContinue)
- else
- begin
- if (CurrentToken=tjsIdentifier) then
- L:=LookupLabel(CurrentTokenString,tjsContinue);
- Consume(tjsIdentifier);
- end;
- Consume(tjsSemicolon,True);
- C.Target:=L.Labelset.Target;
- C.TargetName:=L.Name;
- except
- FreeAndNil(C);
- Raise;
- end;
- end;
- function TJSParser.ParseBreakStatement : TJSElement;
- Var
- L : TJSLabel;
- B : TJSBreakStatement;
- begin
- B:=TJSBreakStatement(CreateElement(TJSBreakStatement));
- try
- Result:=B;
- Consume(tjsBreak);
- If (CurrentToken=tjsSemicolon) then
- L:=LookupLabel(SEmptyLabel,tjsBreak)
- else
- begin
- if (CurrentToken=tjsIdentifier) then
- L:=LookupLabel(CurrentTokenString,tjsBreak);
- Consume(tjsIdentifier);
- end;
- Consume(tjsSemicolon,True);
- B.Target:=L.Labelset.Target;
- B.TargetName:=L.Name;
- except
- FreeAndNil(B);
- Raise;
- end;
- end;
- function TJSParser.ParseReturnStatement : TJSElement;
- Var
- R : TJSreturnStatement;
- begin
- R:=TJSReturnStatement(CreateElement(TJSReturnStatement));
- try
- Result:=R;
- Consume(tjsReturn);
- If (FunctionDepth=0) then
- Error(SErrReturnNotInFunction);
- If Not (CurrentToken in [tjsSemicolon,tjsCurlyBraceClose]) then
- R.Expr:=ParseExpression;
- Consume(tjsSemicolon,True);
- except
- FreeAndNil(R);
- Raise;
- end;
- end;
- function TJSParser.ParseWithStatement : TJSElement;
- Var
- W : TJSWithStatement;
- begin
- W:=TJSWithStatement(CreateElement(TJSWithStatement));
- try
- Consume(tjsWith);
- Consume(tjsBraceOpen);
- W.A:=ParseExpression;
- Consume(tjsBraceClose);
- W.B:=ParseStatement;
- except
- FreeAndNil(W);
- Raise;
- end;
- Result:=W;
- end;
- function TJSParser.ParseSwitchStatement : TJSElement;
- Var
- N : TJSSwitchStatement;
- Ca : TJSCaseElement;
- begin
- N:=TJSSwitchStatement(CreateElement(TJSSwitchStatement));
- try
- N.Target:=CurrentLabelset.Target;
- EnterLabel(SEmptyLabel);
- try
- Consume(tjsSwitch);
- Consume(tjsBraceOpen);
- N.Cond:=ParseExpression;
- Consume(tjsBraceClose);
- Consume(tjsCurlyBraceOpen);
- While (CurrentToken<>tjsCurlyBraceClose) do
- begin
- If (CurrentToken=tjsCase) then
- begin
- GetNextToken;
- Ca:=N.Cases.AddCase;
- Ca.Expr:=ParseExpression;
- end
- else if (CurrentToken=tjsDefault) then
- begin
- If (N.TheDefault<>Nil) then
- Error(SerrDuplicateSwitchDefault);
- Ca:=N.Cases.AddCase;
- N.TheDefault:=Ca;
- GetNextToken;
- end
- else
- Error(SerrCaseEndExpected);
- Consume(tjsColon);
- If Not (CurrentToken in [tjsCurlyBraceClose,tjsCase,tjsDefault]) then
- Ca.Body:=ParseStatementList;
- end;
- Consume(tjsCurlyBraceClose);
- finally
- LeaveLabel;
- FreeCurrentLabelSet;
- end;
- Result:=N;
- except
- FreeAndNil(N);
- Raise;
- end;
- end;
- function TJSParser.ParseThrowStatement : TJSElement;
- Var
- TS : TJSThrowStatement;
- begin
- TS:=TJSThrowStatement(CreateElement(TJSThrowStatement));
- try
- Result:=TS;
- Consume(tjsThrow);
- If IsEndOfLine then
- Error(SErrNewlineAfterThrow);
- TS.A:=ParseExpression;
- Consume(tjsSemicolon,true);
- except
- FreeAndNil(TS);
- Raise;
- end;
- end;
- function TJSParser.ParseTryStatement : TJSElement;
- Var
- BO,BC,BF : TJSElement;
- Id : jstree.TJSString;
- T : TJSTryStatement;
- begin
- BO:=Nil;
- BC:=Nil;
- BF:=Nil;
- Result:=Nil;
- Consume(tjsTry);
- try
- Bo:=ParseBlock;
- if (CurrentToken=tjscatch) then
- begin
- Consume(tjsCatch);
- Consume(tjsBraceOpen);
- if (CurrentToken=tjsIdentifier) then
- id:=CurrentTokenString;
- Consume(tjsIdentifier);
- Consume(tjsBraceClose);
- BC:=ParseBlock;
- end;
- if (CurrentToken=tjsFinally) then
- begin
- consume(tjsFinally);
- BF:=ParseBlock;
- end;
- If (BF=Nil) and (BC=Nil) then
- Error(SErrCatchFinallyExpected);
- If Assigned(BC) AND Assigned(BF) then
- T:=TJSTryStatement(CreateElement(TJSTryCatchFinallyStatement))
- else if Assigned(BC) then
- T:=TJSTryStatement(CreateElement(TJSTryCatchStatement))
- else
- T:=TJSTryStatement(CreateElement(TJSTryFinallyStatement));
- Result:=T;
- T.Block:=Bo;
- Bo:=Nil;
- T.BCatch:=BC;
- BC:=Nil;
- T.BFinally:=BF;
- BF:=Nil;
- T.Ident:=ID;
- except
- FreeAndNil(Bo);
- FreeAndNil(BC);
- FreeAndNil(BF);
- FreeAndNil(Result);
- Raise;
- end;
- end;
- function TJSParser.ParseFunctionExpression : TJSFunctionDeclarationStatement;
- Var
- Oni,olhs: Boolean;
- F : TJSFunctionDeclarationStatement;
- N : String;
- Args : TStrings;
- begin
- {$ifdef debugparser} Writeln('>>> ParseFunctionExpression');{$endif}
- oni:=NoIn;
- olhs:=IsLHS;
- F:=Nil;
- Args:=Nil;
- try
- NoIn:=False;
- IsLHS:=False;
- F:=TJSFunctionDeclarationStatement(CreateElement(TJSFunctionDeclarationStatement));
- try
- Consume(tjsFunction);
- if (CurrentToken=tjsIdentifier) then
- begin
- n:=CurrentTokenstring;
- GetNextToken;
- end
- else
- n:='';
- if n='' then ; // what to do with that?
- Consume(tjsBraceOpen);
- F.AFunction:= TJSFuncDef.Create;
- Args:=ParseFormalParameterList;
- try
- If Assigned(Args) then
- F.AFunction.Params.Assign(Args);
- finally
- FreeAndNil(Args);
- end;
- Consume(tjsBraceClose);
- Consume(tjsCurlyBraceOpen);
- Inc(FFunctionDepth);
- try
- F.AFunction.Body:=ParseFunctionBody;
- Finally
- Dec(FFunctionDepth);
- end;
- Consume(tjsCurlyBraceClose);
- Result:=F;
- except
- FreeAndNil(F);
- Raise;
- end;
- finally
- NoIn := oni;
- IsLHS := olhs;
- end;
- {$ifdef debugparser} Writeln('<<< ParseFunctionExpression');{$endif}
- end;
- function TJSParser.ParseFunctionStatement : TJSElement;
- Var
- F : TJSFunctionDeclarationStatement;
- I : TJSPrimaryExpressionIdent;
- A : TJSAssignStatement;
- E : TJSExpressionStatement;
- begin
- {$ifdef debugparser} Writeln('>>> ParseFunctionStatement');{$endif}
- F:=Nil;
- I:=Nil;
- A:=Nil;
- try
- F:=ParseFunctionExpression;
- I:=TJSPrimaryExpressionIdent(CreateElement(TJSPrimaryExpressionIdent));
- I.Name:=F.AFunction.Name;
- A:=TJSAssignStatement(CreateElement(TJSAssignStatement));
- A.LHS:=I;
- I:=Nil;
- A.Expr:=F;
- F:=Nil;
- E:=TJSExpressionStatement(CreateElement(TJSExpressionStatement));
- E.A:=A;
- A:=Nil;
- Result:=E;
- except
- FreeAndNil(F);
- FreeAndNil(I);
- FreeAndNil(A);
- Raise;
- end;
- {$ifdef debugparser} Writeln('<<< ParseFunctionStatement');{$endif}
- end;
- function TJSParser.ParseLabeledStatement : TJSElement;
- Var
- OL : TJSLabelSet;
- LS : TJSLabeledStatement;
- begin
- LS:=TJSLabeledStatement(CreateElement(TJSLabeledStatement));
- try
- Result:=LS;
- OL:=FCurrentLabelSet;
- try
- FCurrentLabelSet:=Nil;
- LS.target:=CurrentLabelSet.Target;
- Repeat
- LS.TheLabel:=EnterLabel(CurrentTokenString);
- Consume(tjsIdentifier);
- Consume(tjsColon);
- Until (CurrentToken<>tjsIdentifier) or (PeekNextToken<>tjsColon);
- Case CurrentToken of
- tjsDo,tjsWhile,tjsFor : LS.A:=ParseIterationStatement;
- tjsswitch : LS.A:=ParseSwitchStatement;
- else
- LS.A:=ParseStatement;
- end;
- finally
- FreeCurrentLabelSet;
- FCurrentLabelSet:=Ol;
- end;
- except
- FreeAndNil(LS);
- Raise;
- end;
- end;
- procedure TJSParser.FreeCurrentLabelSet;
- Var
- L : TJSLabelSet;
- begin
- While Assigned(FCurrentLabelSet) do
- begin
- L:=FCurrentLabelset.Next;
- FCurrentLabelSet.Free;
- FCurrentLabelSet:=L;
- end;
- end;
- function TJSParser.GetVersion: TECMAVersion;
- begin
- Result:=FSCanner.ECMAVersion;
- end;
- function TJSParser.ParseExpressionStatement : TJSElement;
- Var
- E : TJSElement;
- R : TJSExpressionStatement;
- begin
- {$ifdef debugparser} Writeln('ParseExpressionStatement');{$endif debugparser}
- E:=ParseExpression;
- Consume(tjsSemicolon,True);
- R:=TJSExpressionStatement(CreateElement(TJSExpressionStatement));
- R.A:=E;
- Result:=R;
- {$ifdef debugparser} Writeln('Exit ParseExpressionStatement');{$endif debugparser}
- end;
- function TJSParser.ParseExpression : TJSElement;
- Var
- C : TJSCommaExpression;
- begin
- {$ifdef debugparser} Writeln('ParseExpression');{$endif debugparser}
- Result:=ParseAssignmentExpression;
- try
- If (CurrentToken=tjsComma) then
- begin
- C:=TJSCommaExpression(CreateElement(TJSCommaExpression));
- C.A:=Result;
- Result:=C;
- GetNextToken;
- C.B:=ParseExpression();
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('Exit ParseExpression');{$endif debugparser}
- end;
- function TJSParser.ParseStatement : TJSElement;
- begin
- {$ifdef debugparser} Writeln('>>> Parsestatement');{$endif}
- Result:=Nil;
- Case CurrentToken of
- tjsCurlyBraceOpen :
- Result:=ParseBlock;
- tjsVar:
- Result:=ParseVariableStatement;
- tjsSemicolon:
- Result:=ParseEmptyStatement;
- tjsIf:
- Result:=ParseIfStatement;
- tjsDo,tjsWhile,tjsFor:
- Result:=ParseIterationStatement;
- tjsContinue:
- Result:=ParseContinueStatement;
- tjsBreak:
- Result:=ParseBreakStatement;
- tjsReturn:
- Result:=ParseReturnStatement;
- tjsWith:
- Result:=ParseWithStatement;
- tjsSwitch:
- Result:=ParseSwitchStatement;
- tjsThrow:
- Result:=ParseThrowStatement;
- tjsTry:
- Result:=ParseTryStatement;
- tjsFunction:
- begin
- If (PeekNextToken<>tjsBraceOpen) then
- Result:=ParseFunctionStatement;
- Error(SErrFunctionNotAllowedHere);
- end;
- tjsIdentifier:
- If (PeekNextToken=tjsColon) then
- Result:=ParseLabeledStatement
- else
- Result:=ParseExpressionStatement;
- else
- Result:=ParseExpressionStatement;
- end;
- {$ifdef debugparser} If Assigned(Result) then Writeln('<<< Parsestatement ',Result.ClassName) else Writeln('<<< Parsestatement (null');{$endif}
- end;
- function TJSParser.ParseSourceElements : TJSSourceElements;
- Const
- StatementTokens = [tjsNULL, tjsTRUE, tjsFALSE,
- tjsTHIS, tjsIdentifier,jstoken.tjsSTRING,tjsNUMBER,
- tjsBraceOpen,tjsCurlyBraceOpen,tjsSquaredBraceOpen,
- tjsNew,tjsDelete,tjsVoid,tjsTypeOf,
- tjsPlusPlus,tjsMinusMinus,
- tjsPlus,tjsMinus,tjsNot,tjsNE,tjsSNE,tjsSemicolon,
- tjsVAR,tjsIF,tjsDO,tjsWHILE,tjsFOR,jstoken.tjsCONTINUE,jstoken.tjsBREAK,jstoken.tjsReturn,
- tjsWith,jstoken.tjsSWITCH,tjsThrow,TjsTry,tjsDIV,tjsDIVEQ];
- Var
- F : TJSFunctionDeclarationStatement;
- E : TJSElement;
- Done : Boolean;
- VS : TJSElementNodes;
- aSync : Boolean;
- begin
- {$ifdef debugparser} Writeln('>>> Entering source elements');{$endif}
- Result:=TJSSourceElements(CreateElement(TJSSourceElements));
- try
- Done:=False;
- aSync:=False;
- VS:=FCurrentVars;
- Try
- FCurrentVars:=Result.Vars;
- Repeat
- {$ifdef debugparser} Writeln('Sourceelements start:',GetEnumName(TypeInfo(TJSToken),Ord(CurrentToken)), ' As string: ',CurrentTokenString);{$endif debugparser}
- aSync:= (ECMAVersion>=MinAsyncVersion) and (CurrentToken=tjsIdentifier) and (CurrentTokenString='async');
- if aSync then
- GetNextToken;
- If (CurrentToken=jstoken.tjsFunction) then
- begin
- If (PeekNextToken<>tjsBraceOpen) then
- begin
- F:=Self.ParseFunctionDeclaration;
- F.AFunction.IsAsync:=aSync;
- Result.Functions.AddNode.Node:=F;
- end
- else
- begin
- {$ifdef debugparser} Writeln('Function expression detected');{$endif}
- E:=Self.ParseStatement;
- Result.Statements.AddNode.Node:=E;
- end;
- end
- else if CurrentToken in StatementTokens then
- begin
- E:=Self.ParseStatement;
- Result.Statements.AddNode.Node:=E;
- end
- else
- Done:=True;
- {$ifdef debugparser} Writeln('Sourceelements Done : ',Done);{$endif}
- Until Done;
- Finally
- FCurrentVars:=VS;
- end;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('<<< Exiting source elements');{$endif}
- end;
- function TJSParser.ParseFunctionBody : TJSFunctionBody;
- Var
- E : TJSElement;
- begin
- {$ifdef debugparser} Writeln('>>> Entering FunctionBody');{$endif}
- Result:=TJSFunctionBody(CreateElement(TJSFunctionBody));
- try
- E:=Self.ParseSourceElements;
- Result.A:=E;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('<<< Exiting FunctionBody');{$endif}
- end;
- Function TJSParser.ParseProgram: TJSFunctionDeclarationStatement;
- Var
- B : TJSElement;
- begin
- {$ifdef debugparser} Writeln('>>> Entering FunctionDeclarationStatement');{$endif}
- B:=Parse;
- If Not (B is TJSFunctionBody) then
- Error('Parse did not result in functionbody');
- Result:=TJSFunctionDeclarationStatement(CreateElement(TJSFunctionDeclarationStatement));
- Result.AFunction:=TJSFuncDef.Create;
- Result.AFunction.Body:=TJSFunctionBody(B);
- {$ifdef debugparser} Writeln('<<< Exiting FunctionDeclarationStatement');{$endif}
- end;
- Function TJSParser.Parse: TJSElement;
- Var
- Body : TJSElement;
- begin
- {$ifdef debugparser} Writeln('>>> Parse');{$endif}
- Result:=Nil;
- CheckParser;
- GetNextToken;
- Body:=ParseFunctionBody;
- Result:=Body;
- try
- if (CurrentToken<>tjsEOF) then
- begin
- if (CurrentToken=tjsCurlyBraceClose) then
- Error(SErrUnmatchedCurlyBrace)
- else if (CurrentToken=tjsBraceClose) then
- Error(SerrUnmatchedBrace)
- else if (CurrentToken=tjsSquaredBraceClose) then
- Error(SerrUnmatchedSquareBrace);
- Error(SErrUnexpectedToken,[CurrentTokenString]);
- end;
- If (Body is TJSFunctionBody) then
- TJSFunctionBody(Body).isProgram:=True;
- except
- FreeAndNil(Result);
- Raise;
- end;
- {$ifdef debugparser} Writeln('<<< Parse');{$endif}
- end;
- end.
|