tcmodules.pas 862 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804980598069807980898099810981198129813981498159816981798189819982098219822982398249825982698279828982998309831983298339834983598369837983898399840984198429843984498459846984798489849985098519852985398549855985698579858985998609861986298639864986598669867986898699870987198729873987498759876987798789879988098819882988398849885988698879888988998909891989298939894989598969897989898999900990199029903990499059906990799089909991099119912991399149915991699179918991999209921992299239924992599269927992899299930993199329933993499359936993799389939994099419942994399449945994699479948994999509951995299539954995599569957995899599960996199629963996499659966996799689969997099719972997399749975997699779978997999809981998299839984998599869987998899899990999199929993999499959996999799989999100001000110002100031000410005100061000710008100091001010011100121001310014100151001610017100181001910020100211002210023100241002510026100271002810029100301003110032100331003410035100361003710038100391004010041100421004310044100451004610047100481004910050100511005210053100541005510056100571005810059100601006110062100631006410065100661006710068100691007010071100721007310074100751007610077100781007910080100811008210083100841008510086100871008810089100901009110092100931009410095100961009710098100991010010101101021010310104101051010610107101081010910110101111011210113101141011510116101171011810119101201012110122101231012410125101261012710128101291013010131101321013310134101351013610137101381013910140101411014210143101441014510146101471014810149101501015110152101531015410155101561015710158101591016010161101621016310164101651016610167101681016910170101711017210173101741017510176101771017810179101801018110182101831018410185101861018710188101891019010191101921019310194101951019610197101981019910200102011020210203102041020510206102071020810209102101021110212102131021410215102161021710218102191022010221102221022310224102251022610227102281022910230102311023210233102341023510236102371023810239102401024110242102431024410245102461024710248102491025010251102521025310254102551025610257102581025910260102611026210263102641026510266102671026810269102701027110272102731027410275102761027710278102791028010281102821028310284102851028610287102881028910290102911029210293102941029510296102971029810299103001030110302103031030410305103061030710308103091031010311103121031310314103151031610317103181031910320103211032210323103241032510326103271032810329103301033110332103331033410335103361033710338103391034010341103421034310344103451034610347103481034910350103511035210353103541035510356103571035810359103601036110362103631036410365103661036710368103691037010371103721037310374103751037610377103781037910380103811038210383103841038510386103871038810389103901039110392103931039410395103961039710398103991040010401104021040310404104051040610407104081040910410104111041210413104141041510416104171041810419104201042110422104231042410425104261042710428104291043010431104321043310434104351043610437104381043910440104411044210443104441044510446104471044810449104501045110452104531045410455104561045710458104591046010461104621046310464104651046610467104681046910470104711047210473104741047510476104771047810479104801048110482104831048410485104861048710488104891049010491104921049310494104951049610497104981049910500105011050210503105041050510506105071050810509105101051110512105131051410515105161051710518105191052010521105221052310524105251052610527105281052910530105311053210533105341053510536105371053810539105401054110542105431054410545105461054710548105491055010551105521055310554105551055610557105581055910560105611056210563105641056510566105671056810569105701057110572105731057410575105761057710578105791058010581105821058310584105851058610587105881058910590105911059210593105941059510596105971059810599106001060110602106031060410605106061060710608106091061010611106121061310614106151061610617106181061910620106211062210623106241062510626106271062810629106301063110632106331063410635106361063710638106391064010641106421064310644106451064610647106481064910650106511065210653106541065510656106571065810659106601066110662106631066410665106661066710668106691067010671106721067310674106751067610677106781067910680106811068210683106841068510686106871068810689106901069110692106931069410695106961069710698106991070010701107021070310704107051070610707107081070910710107111071210713107141071510716107171071810719107201072110722107231072410725107261072710728107291073010731107321073310734107351073610737107381073910740107411074210743107441074510746107471074810749107501075110752107531075410755107561075710758107591076010761107621076310764107651076610767107681076910770107711077210773107741077510776107771077810779107801078110782107831078410785107861078710788107891079010791107921079310794107951079610797107981079910800108011080210803108041080510806108071080810809108101081110812108131081410815108161081710818108191082010821108221082310824108251082610827108281082910830108311083210833108341083510836108371083810839108401084110842108431084410845108461084710848108491085010851108521085310854108551085610857108581085910860108611086210863108641086510866108671086810869108701087110872108731087410875108761087710878108791088010881108821088310884108851088610887108881088910890108911089210893108941089510896108971089810899109001090110902109031090410905109061090710908109091091010911109121091310914109151091610917109181091910920109211092210923109241092510926109271092810929109301093110932109331093410935109361093710938109391094010941109421094310944109451094610947109481094910950109511095210953109541095510956109571095810959109601096110962109631096410965109661096710968109691097010971109721097310974109751097610977109781097910980109811098210983109841098510986109871098810989109901099110992109931099410995109961099710998109991100011001110021100311004110051100611007110081100911010110111101211013110141101511016110171101811019110201102111022110231102411025110261102711028110291103011031110321103311034110351103611037110381103911040110411104211043110441104511046110471104811049110501105111052110531105411055110561105711058110591106011061110621106311064110651106611067110681106911070110711107211073110741107511076110771107811079110801108111082110831108411085110861108711088110891109011091110921109311094110951109611097110981109911100111011110211103111041110511106111071110811109111101111111112111131111411115111161111711118111191112011121111221112311124111251112611127111281112911130111311113211133111341113511136111371113811139111401114111142111431114411145111461114711148111491115011151111521115311154111551115611157111581115911160111611116211163111641116511166111671116811169111701117111172111731117411175111761117711178111791118011181111821118311184111851118611187111881118911190111911119211193111941119511196111971119811199112001120111202112031120411205112061120711208112091121011211112121121311214112151121611217112181121911220112211122211223112241122511226112271122811229112301123111232112331123411235112361123711238112391124011241112421124311244112451124611247112481124911250112511125211253112541125511256112571125811259112601126111262112631126411265112661126711268112691127011271112721127311274112751127611277112781127911280112811128211283112841128511286112871128811289112901129111292112931129411295112961129711298112991130011301113021130311304113051130611307113081130911310113111131211313113141131511316113171131811319113201132111322113231132411325113261132711328113291133011331113321133311334113351133611337113381133911340113411134211343113441134511346113471134811349113501135111352113531135411355113561135711358113591136011361113621136311364113651136611367113681136911370113711137211373113741137511376113771137811379113801138111382113831138411385113861138711388113891139011391113921139311394113951139611397113981139911400114011140211403114041140511406114071140811409114101141111412114131141411415114161141711418114191142011421114221142311424114251142611427114281142911430114311143211433114341143511436114371143811439114401144111442114431144411445114461144711448114491145011451114521145311454114551145611457114581145911460114611146211463114641146511466114671146811469114701147111472114731147411475114761147711478114791148011481114821148311484114851148611487114881148911490114911149211493114941149511496114971149811499115001150111502115031150411505115061150711508115091151011511115121151311514115151151611517115181151911520115211152211523115241152511526115271152811529115301153111532115331153411535115361153711538115391154011541115421154311544115451154611547115481154911550115511155211553115541155511556115571155811559115601156111562115631156411565115661156711568115691157011571115721157311574115751157611577115781157911580115811158211583115841158511586115871158811589115901159111592115931159411595115961159711598115991160011601116021160311604116051160611607116081160911610116111161211613116141161511616116171161811619116201162111622116231162411625116261162711628116291163011631116321163311634116351163611637116381163911640116411164211643116441164511646116471164811649116501165111652116531165411655116561165711658116591166011661116621166311664116651166611667116681166911670116711167211673116741167511676116771167811679116801168111682116831168411685116861168711688116891169011691116921169311694116951169611697116981169911700117011170211703117041170511706117071170811709117101171111712117131171411715117161171711718117191172011721117221172311724117251172611727117281172911730117311173211733117341173511736117371173811739117401174111742117431174411745117461174711748117491175011751117521175311754117551175611757117581175911760117611176211763117641176511766117671176811769117701177111772117731177411775117761177711778117791178011781117821178311784117851178611787117881178911790117911179211793117941179511796117971179811799118001180111802118031180411805118061180711808118091181011811118121181311814118151181611817118181181911820118211182211823118241182511826118271182811829118301183111832118331183411835118361183711838118391184011841118421184311844118451184611847118481184911850118511185211853118541185511856118571185811859118601186111862118631186411865118661186711868118691187011871118721187311874118751187611877118781187911880118811188211883118841188511886118871188811889118901189111892118931189411895118961189711898118991190011901119021190311904119051190611907119081190911910119111191211913119141191511916119171191811919119201192111922119231192411925119261192711928119291193011931119321193311934119351193611937119381193911940119411194211943119441194511946119471194811949119501195111952119531195411955119561195711958119591196011961119621196311964119651196611967119681196911970119711197211973119741197511976119771197811979119801198111982119831198411985119861198711988119891199011991119921199311994119951199611997119981199912000120011200212003120041200512006120071200812009120101201112012120131201412015120161201712018120191202012021120221202312024120251202612027120281202912030120311203212033120341203512036120371203812039120401204112042120431204412045120461204712048120491205012051120521205312054120551205612057120581205912060120611206212063120641206512066120671206812069120701207112072120731207412075120761207712078120791208012081120821208312084120851208612087120881208912090120911209212093120941209512096120971209812099121001210112102121031210412105121061210712108121091211012111121121211312114121151211612117121181211912120121211212212123121241212512126121271212812129121301213112132121331213412135121361213712138121391214012141121421214312144121451214612147121481214912150121511215212153121541215512156121571215812159121601216112162121631216412165121661216712168121691217012171121721217312174121751217612177121781217912180121811218212183121841218512186121871218812189121901219112192121931219412195121961219712198121991220012201122021220312204122051220612207122081220912210122111221212213122141221512216122171221812219122201222112222122231222412225122261222712228122291223012231122321223312234122351223612237122381223912240122411224212243122441224512246122471224812249122501225112252122531225412255122561225712258122591226012261122621226312264122651226612267122681226912270122711227212273122741227512276122771227812279122801228112282122831228412285122861228712288122891229012291122921229312294122951229612297122981229912300123011230212303123041230512306123071230812309123101231112312123131231412315123161231712318123191232012321123221232312324123251232612327123281232912330123311233212333123341233512336123371233812339123401234112342123431234412345123461234712348123491235012351123521235312354123551235612357123581235912360123611236212363123641236512366123671236812369123701237112372123731237412375123761237712378123791238012381123821238312384123851238612387123881238912390123911239212393123941239512396123971239812399124001240112402124031240412405124061240712408124091241012411124121241312414124151241612417124181241912420124211242212423124241242512426124271242812429124301243112432124331243412435124361243712438124391244012441124421244312444124451244612447124481244912450124511245212453124541245512456124571245812459124601246112462124631246412465124661246712468124691247012471124721247312474124751247612477124781247912480124811248212483124841248512486124871248812489124901249112492124931249412495124961249712498124991250012501125021250312504125051250612507125081250912510125111251212513125141251512516125171251812519125201252112522125231252412525125261252712528125291253012531125321253312534125351253612537125381253912540125411254212543125441254512546125471254812549125501255112552125531255412555125561255712558125591256012561125621256312564125651256612567125681256912570125711257212573125741257512576125771257812579125801258112582125831258412585125861258712588125891259012591125921259312594125951259612597125981259912600126011260212603126041260512606126071260812609126101261112612126131261412615126161261712618126191262012621126221262312624126251262612627126281262912630126311263212633126341263512636126371263812639126401264112642126431264412645126461264712648126491265012651126521265312654126551265612657126581265912660126611266212663126641266512666126671266812669126701267112672126731267412675126761267712678126791268012681126821268312684126851268612687126881268912690126911269212693126941269512696126971269812699127001270112702127031270412705127061270712708127091271012711127121271312714127151271612717127181271912720127211272212723127241272512726127271272812729127301273112732127331273412735127361273712738127391274012741127421274312744127451274612747127481274912750127511275212753127541275512756127571275812759127601276112762127631276412765127661276712768127691277012771127721277312774127751277612777127781277912780127811278212783127841278512786127871278812789127901279112792127931279412795127961279712798127991280012801128021280312804128051280612807128081280912810128111281212813128141281512816128171281812819128201282112822128231282412825128261282712828128291283012831128321283312834128351283612837128381283912840128411284212843128441284512846128471284812849128501285112852128531285412855128561285712858128591286012861128621286312864128651286612867128681286912870128711287212873128741287512876128771287812879128801288112882128831288412885128861288712888128891289012891128921289312894128951289612897128981289912900129011290212903129041290512906129071290812909129101291112912129131291412915129161291712918129191292012921129221292312924129251292612927129281292912930129311293212933129341293512936129371293812939129401294112942129431294412945129461294712948129491295012951129521295312954129551295612957129581295912960129611296212963129641296512966129671296812969129701297112972129731297412975129761297712978129791298012981129821298312984129851298612987129881298912990129911299212993129941299512996129971299812999130001300113002130031300413005130061300713008130091301013011130121301313014130151301613017130181301913020130211302213023130241302513026130271302813029130301303113032130331303413035130361303713038130391304013041130421304313044130451304613047130481304913050130511305213053130541305513056130571305813059130601306113062130631306413065130661306713068130691307013071130721307313074130751307613077130781307913080130811308213083130841308513086130871308813089130901309113092130931309413095130961309713098130991310013101131021310313104131051310613107131081310913110131111311213113131141311513116131171311813119131201312113122131231312413125131261312713128131291313013131131321313313134131351313613137131381313913140131411314213143131441314513146131471314813149131501315113152131531315413155131561315713158131591316013161131621316313164131651316613167131681316913170131711317213173131741317513176131771317813179131801318113182131831318413185131861318713188131891319013191131921319313194131951319613197131981319913200132011320213203132041320513206132071320813209132101321113212132131321413215132161321713218132191322013221132221322313224132251322613227132281322913230132311323213233132341323513236132371323813239132401324113242132431324413245132461324713248132491325013251132521325313254132551325613257132581325913260132611326213263132641326513266132671326813269132701327113272132731327413275132761327713278132791328013281132821328313284132851328613287132881328913290132911329213293132941329513296132971329813299133001330113302133031330413305133061330713308133091331013311133121331313314133151331613317133181331913320133211332213323133241332513326133271332813329133301333113332133331333413335133361333713338133391334013341133421334313344133451334613347133481334913350133511335213353133541335513356133571335813359133601336113362133631336413365133661336713368133691337013371133721337313374133751337613377133781337913380133811338213383133841338513386133871338813389133901339113392133931339413395133961339713398133991340013401134021340313404134051340613407134081340913410134111341213413134141341513416134171341813419134201342113422134231342413425134261342713428134291343013431134321343313434134351343613437134381343913440134411344213443134441344513446134471344813449134501345113452134531345413455134561345713458134591346013461134621346313464134651346613467134681346913470134711347213473134741347513476134771347813479134801348113482134831348413485134861348713488134891349013491134921349313494134951349613497134981349913500135011350213503135041350513506135071350813509135101351113512135131351413515135161351713518135191352013521135221352313524135251352613527135281352913530135311353213533135341353513536135371353813539135401354113542135431354413545135461354713548135491355013551135521355313554135551355613557135581355913560135611356213563135641356513566135671356813569135701357113572135731357413575135761357713578135791358013581135821358313584135851358613587135881358913590135911359213593135941359513596135971359813599136001360113602136031360413605136061360713608136091361013611136121361313614136151361613617136181361913620136211362213623136241362513626136271362813629136301363113632136331363413635136361363713638136391364013641136421364313644136451364613647136481364913650136511365213653136541365513656136571365813659136601366113662136631366413665136661366713668136691367013671136721367313674136751367613677136781367913680136811368213683136841368513686136871368813689136901369113692136931369413695136961369713698136991370013701137021370313704137051370613707137081370913710137111371213713137141371513716137171371813719137201372113722137231372413725137261372713728137291373013731137321373313734137351373613737137381373913740137411374213743137441374513746137471374813749137501375113752137531375413755137561375713758137591376013761137621376313764137651376613767137681376913770137711377213773137741377513776137771377813779137801378113782137831378413785137861378713788137891379013791137921379313794137951379613797137981379913800138011380213803138041380513806138071380813809138101381113812138131381413815138161381713818138191382013821138221382313824138251382613827138281382913830138311383213833138341383513836138371383813839138401384113842138431384413845138461384713848138491385013851138521385313854138551385613857138581385913860138611386213863138641386513866138671386813869138701387113872138731387413875138761387713878138791388013881138821388313884138851388613887138881388913890138911389213893138941389513896138971389813899139001390113902139031390413905139061390713908139091391013911139121391313914139151391613917139181391913920139211392213923139241392513926139271392813929139301393113932139331393413935139361393713938139391394013941139421394313944139451394613947139481394913950139511395213953139541395513956139571395813959139601396113962139631396413965139661396713968139691397013971139721397313974139751397613977139781397913980139811398213983139841398513986139871398813989139901399113992139931399413995139961399713998139991400014001140021400314004140051400614007140081400914010140111401214013140141401514016140171401814019140201402114022140231402414025140261402714028140291403014031140321403314034140351403614037140381403914040140411404214043140441404514046140471404814049140501405114052140531405414055140561405714058140591406014061140621406314064140651406614067140681406914070140711407214073140741407514076140771407814079140801408114082140831408414085140861408714088140891409014091140921409314094140951409614097140981409914100141011410214103141041410514106141071410814109141101411114112141131411414115141161411714118141191412014121141221412314124141251412614127141281412914130141311413214133141341413514136141371413814139141401414114142141431414414145141461414714148141491415014151141521415314154141551415614157141581415914160141611416214163141641416514166141671416814169141701417114172141731417414175141761417714178141791418014181141821418314184141851418614187141881418914190141911419214193141941419514196141971419814199142001420114202142031420414205142061420714208142091421014211142121421314214142151421614217142181421914220142211422214223142241422514226142271422814229142301423114232142331423414235142361423714238142391424014241142421424314244142451424614247142481424914250142511425214253142541425514256142571425814259142601426114262142631426414265142661426714268142691427014271142721427314274142751427614277142781427914280142811428214283142841428514286142871428814289142901429114292142931429414295142961429714298142991430014301143021430314304143051430614307143081430914310143111431214313143141431514316143171431814319143201432114322143231432414325143261432714328143291433014331143321433314334143351433614337143381433914340143411434214343143441434514346143471434814349143501435114352143531435414355143561435714358143591436014361143621436314364143651436614367143681436914370143711437214373143741437514376143771437814379143801438114382143831438414385143861438714388143891439014391143921439314394143951439614397143981439914400144011440214403144041440514406144071440814409144101441114412144131441414415144161441714418144191442014421144221442314424144251442614427144281442914430144311443214433144341443514436144371443814439144401444114442144431444414445144461444714448144491445014451144521445314454144551445614457144581445914460144611446214463144641446514466144671446814469144701447114472144731447414475144761447714478144791448014481144821448314484144851448614487144881448914490144911449214493144941449514496144971449814499145001450114502145031450414505145061450714508145091451014511145121451314514145151451614517145181451914520145211452214523145241452514526145271452814529145301453114532145331453414535145361453714538145391454014541145421454314544145451454614547145481454914550145511455214553145541455514556145571455814559145601456114562145631456414565145661456714568145691457014571145721457314574145751457614577145781457914580145811458214583145841458514586145871458814589145901459114592145931459414595145961459714598145991460014601146021460314604146051460614607146081460914610146111461214613146141461514616146171461814619146201462114622146231462414625146261462714628146291463014631146321463314634146351463614637146381463914640146411464214643146441464514646146471464814649146501465114652146531465414655146561465714658146591466014661146621466314664146651466614667146681466914670146711467214673146741467514676146771467814679146801468114682146831468414685146861468714688146891469014691146921469314694146951469614697146981469914700147011470214703147041470514706147071470814709147101471114712147131471414715147161471714718147191472014721147221472314724147251472614727147281472914730147311473214733147341473514736147371473814739147401474114742147431474414745147461474714748147491475014751147521475314754147551475614757147581475914760147611476214763147641476514766147671476814769147701477114772147731477414775147761477714778147791478014781147821478314784147851478614787147881478914790147911479214793147941479514796147971479814799148001480114802148031480414805148061480714808148091481014811148121481314814148151481614817148181481914820148211482214823148241482514826148271482814829148301483114832148331483414835148361483714838148391484014841148421484314844148451484614847148481484914850148511485214853148541485514856148571485814859148601486114862148631486414865148661486714868148691487014871148721487314874148751487614877148781487914880148811488214883148841488514886148871488814889148901489114892148931489414895148961489714898148991490014901149021490314904149051490614907149081490914910149111491214913149141491514916149171491814919149201492114922149231492414925149261492714928149291493014931149321493314934149351493614937149381493914940149411494214943149441494514946149471494814949149501495114952149531495414955149561495714958149591496014961149621496314964149651496614967149681496914970149711497214973149741497514976149771497814979149801498114982149831498414985149861498714988149891499014991149921499314994149951499614997149981499915000150011500215003150041500515006150071500815009150101501115012150131501415015150161501715018150191502015021150221502315024150251502615027150281502915030150311503215033150341503515036150371503815039150401504115042150431504415045150461504715048150491505015051150521505315054150551505615057150581505915060150611506215063150641506515066150671506815069150701507115072150731507415075150761507715078150791508015081150821508315084150851508615087150881508915090150911509215093150941509515096150971509815099151001510115102151031510415105151061510715108151091511015111151121511315114151151511615117151181511915120151211512215123151241512515126151271512815129151301513115132151331513415135151361513715138151391514015141151421514315144151451514615147151481514915150151511515215153151541515515156151571515815159151601516115162151631516415165151661516715168151691517015171151721517315174151751517615177151781517915180151811518215183151841518515186151871518815189151901519115192151931519415195151961519715198151991520015201152021520315204152051520615207152081520915210152111521215213152141521515216152171521815219152201522115222152231522415225152261522715228152291523015231152321523315234152351523615237152381523915240152411524215243152441524515246152471524815249152501525115252152531525415255152561525715258152591526015261152621526315264152651526615267152681526915270152711527215273152741527515276152771527815279152801528115282152831528415285152861528715288152891529015291152921529315294152951529615297152981529915300153011530215303153041530515306153071530815309153101531115312153131531415315153161531715318153191532015321153221532315324153251532615327153281532915330153311533215333153341533515336153371533815339153401534115342153431534415345153461534715348153491535015351153521535315354153551535615357153581535915360153611536215363153641536515366153671536815369153701537115372153731537415375153761537715378153791538015381153821538315384153851538615387153881538915390153911539215393153941539515396153971539815399154001540115402154031540415405154061540715408154091541015411154121541315414154151541615417154181541915420154211542215423154241542515426154271542815429154301543115432154331543415435154361543715438154391544015441154421544315444154451544615447154481544915450154511545215453154541545515456154571545815459154601546115462154631546415465154661546715468154691547015471154721547315474154751547615477154781547915480154811548215483154841548515486154871548815489154901549115492154931549415495154961549715498154991550015501155021550315504155051550615507155081550915510155111551215513155141551515516155171551815519155201552115522155231552415525155261552715528155291553015531155321553315534155351553615537155381553915540155411554215543155441554515546155471554815549155501555115552155531555415555155561555715558155591556015561155621556315564155651556615567155681556915570155711557215573155741557515576155771557815579155801558115582155831558415585155861558715588155891559015591155921559315594155951559615597155981559915600156011560215603156041560515606156071560815609156101561115612156131561415615156161561715618156191562015621156221562315624156251562615627156281562915630156311563215633156341563515636156371563815639156401564115642156431564415645156461564715648156491565015651156521565315654156551565615657156581565915660156611566215663156641566515666156671566815669156701567115672156731567415675156761567715678156791568015681156821568315684156851568615687156881568915690156911569215693156941569515696156971569815699157001570115702157031570415705157061570715708157091571015711157121571315714157151571615717157181571915720157211572215723157241572515726157271572815729157301573115732157331573415735157361573715738157391574015741157421574315744157451574615747157481574915750157511575215753157541575515756157571575815759157601576115762157631576415765157661576715768157691577015771157721577315774157751577615777157781577915780157811578215783157841578515786157871578815789157901579115792157931579415795157961579715798157991580015801158021580315804158051580615807158081580915810158111581215813158141581515816158171581815819158201582115822158231582415825158261582715828158291583015831158321583315834158351583615837158381583915840158411584215843158441584515846158471584815849158501585115852158531585415855158561585715858158591586015861158621586315864158651586615867158681586915870158711587215873158741587515876158771587815879158801588115882158831588415885158861588715888158891589015891158921589315894158951589615897158981589915900159011590215903159041590515906159071590815909159101591115912159131591415915159161591715918159191592015921159221592315924159251592615927159281592915930159311593215933159341593515936159371593815939159401594115942159431594415945159461594715948159491595015951159521595315954159551595615957159581595915960159611596215963159641596515966159671596815969159701597115972159731597415975159761597715978159791598015981159821598315984159851598615987159881598915990159911599215993159941599515996159971599815999160001600116002160031600416005160061600716008160091601016011160121601316014160151601616017160181601916020160211602216023160241602516026160271602816029160301603116032160331603416035160361603716038160391604016041160421604316044160451604616047160481604916050160511605216053160541605516056160571605816059160601606116062160631606416065160661606716068160691607016071160721607316074160751607616077160781607916080160811608216083160841608516086160871608816089160901609116092160931609416095160961609716098160991610016101161021610316104161051610616107161081610916110161111611216113161141611516116161171611816119161201612116122161231612416125161261612716128161291613016131161321613316134161351613616137161381613916140161411614216143161441614516146161471614816149161501615116152161531615416155161561615716158161591616016161161621616316164161651616616167161681616916170161711617216173161741617516176161771617816179161801618116182161831618416185161861618716188161891619016191161921619316194161951619616197161981619916200162011620216203162041620516206162071620816209162101621116212162131621416215162161621716218162191622016221162221622316224162251622616227162281622916230162311623216233162341623516236162371623816239162401624116242162431624416245162461624716248162491625016251162521625316254162551625616257162581625916260162611626216263162641626516266162671626816269162701627116272162731627416275162761627716278162791628016281162821628316284162851628616287162881628916290162911629216293162941629516296162971629816299163001630116302163031630416305163061630716308163091631016311163121631316314163151631616317163181631916320163211632216323163241632516326163271632816329163301633116332163331633416335163361633716338163391634016341163421634316344163451634616347163481634916350163511635216353163541635516356163571635816359163601636116362163631636416365163661636716368163691637016371163721637316374163751637616377163781637916380163811638216383163841638516386163871638816389163901639116392163931639416395163961639716398163991640016401164021640316404164051640616407164081640916410164111641216413164141641516416164171641816419164201642116422164231642416425164261642716428164291643016431164321643316434164351643616437164381643916440164411644216443164441644516446164471644816449164501645116452164531645416455164561645716458164591646016461164621646316464164651646616467164681646916470164711647216473164741647516476164771647816479164801648116482164831648416485164861648716488164891649016491164921649316494164951649616497164981649916500165011650216503165041650516506165071650816509165101651116512165131651416515165161651716518165191652016521165221652316524165251652616527165281652916530165311653216533165341653516536165371653816539165401654116542165431654416545165461654716548165491655016551165521655316554165551655616557165581655916560165611656216563165641656516566165671656816569165701657116572165731657416575165761657716578165791658016581165821658316584165851658616587165881658916590165911659216593165941659516596165971659816599166001660116602166031660416605166061660716608166091661016611166121661316614166151661616617166181661916620166211662216623166241662516626166271662816629166301663116632166331663416635166361663716638166391664016641166421664316644166451664616647166481664916650166511665216653166541665516656166571665816659166601666116662166631666416665166661666716668166691667016671166721667316674166751667616677166781667916680166811668216683166841668516686166871668816689166901669116692166931669416695166961669716698166991670016701167021670316704167051670616707167081670916710167111671216713167141671516716167171671816719167201672116722167231672416725167261672716728167291673016731167321673316734167351673616737167381673916740167411674216743167441674516746167471674816749167501675116752167531675416755167561675716758167591676016761167621676316764167651676616767167681676916770167711677216773167741677516776167771677816779167801678116782167831678416785167861678716788167891679016791167921679316794167951679616797167981679916800168011680216803168041680516806168071680816809168101681116812168131681416815168161681716818168191682016821168221682316824168251682616827168281682916830168311683216833168341683516836168371683816839168401684116842168431684416845168461684716848168491685016851168521685316854168551685616857168581685916860168611686216863168641686516866168671686816869168701687116872168731687416875168761687716878168791688016881168821688316884168851688616887168881688916890168911689216893168941689516896168971689816899169001690116902169031690416905169061690716908169091691016911169121691316914169151691616917169181691916920169211692216923169241692516926169271692816929169301693116932169331693416935169361693716938169391694016941169421694316944169451694616947169481694916950169511695216953169541695516956169571695816959169601696116962169631696416965169661696716968169691697016971169721697316974169751697616977169781697916980169811698216983169841698516986169871698816989169901699116992169931699416995169961699716998169991700017001170021700317004170051700617007170081700917010170111701217013170141701517016170171701817019170201702117022170231702417025170261702717028170291703017031170321703317034170351703617037170381703917040170411704217043170441704517046170471704817049170501705117052170531705417055170561705717058170591706017061170621706317064170651706617067170681706917070170711707217073170741707517076170771707817079170801708117082170831708417085170861708717088170891709017091170921709317094170951709617097170981709917100171011710217103171041710517106171071710817109171101711117112171131711417115171161711717118171191712017121171221712317124171251712617127171281712917130171311713217133171341713517136171371713817139171401714117142171431714417145171461714717148171491715017151171521715317154171551715617157171581715917160171611716217163171641716517166171671716817169171701717117172171731717417175171761717717178171791718017181171821718317184171851718617187171881718917190171911719217193171941719517196171971719817199172001720117202172031720417205172061720717208172091721017211172121721317214172151721617217172181721917220172211722217223172241722517226172271722817229172301723117232172331723417235172361723717238172391724017241172421724317244172451724617247172481724917250172511725217253172541725517256172571725817259172601726117262172631726417265172661726717268172691727017271172721727317274172751727617277172781727917280172811728217283172841728517286172871728817289172901729117292172931729417295172961729717298172991730017301173021730317304173051730617307173081730917310173111731217313173141731517316173171731817319173201732117322173231732417325173261732717328173291733017331173321733317334173351733617337173381733917340173411734217343173441734517346173471734817349173501735117352173531735417355173561735717358173591736017361173621736317364173651736617367173681736917370173711737217373173741737517376173771737817379173801738117382173831738417385173861738717388173891739017391173921739317394173951739617397173981739917400174011740217403174041740517406174071740817409174101741117412174131741417415174161741717418174191742017421174221742317424174251742617427174281742917430174311743217433174341743517436174371743817439174401744117442174431744417445174461744717448174491745017451174521745317454174551745617457174581745917460174611746217463174641746517466174671746817469174701747117472174731747417475174761747717478174791748017481174821748317484174851748617487174881748917490174911749217493174941749517496174971749817499175001750117502175031750417505175061750717508175091751017511175121751317514175151751617517175181751917520175211752217523175241752517526175271752817529175301753117532175331753417535175361753717538175391754017541175421754317544175451754617547175481754917550175511755217553175541755517556175571755817559175601756117562175631756417565175661756717568175691757017571175721757317574175751757617577175781757917580175811758217583175841758517586175871758817589175901759117592175931759417595175961759717598175991760017601176021760317604176051760617607176081760917610176111761217613176141761517616176171761817619176201762117622176231762417625176261762717628176291763017631176321763317634176351763617637176381763917640176411764217643176441764517646176471764817649176501765117652176531765417655176561765717658176591766017661176621766317664176651766617667176681766917670176711767217673176741767517676176771767817679176801768117682176831768417685176861768717688176891769017691176921769317694176951769617697176981769917700177011770217703177041770517706177071770817709177101771117712177131771417715177161771717718177191772017721177221772317724177251772617727177281772917730177311773217733177341773517736177371773817739177401774117742177431774417745177461774717748177491775017751177521775317754177551775617757177581775917760177611776217763177641776517766177671776817769177701777117772177731777417775177761777717778177791778017781177821778317784177851778617787177881778917790177911779217793177941779517796177971779817799178001780117802178031780417805178061780717808178091781017811178121781317814178151781617817178181781917820178211782217823178241782517826178271782817829178301783117832178331783417835178361783717838178391784017841178421784317844178451784617847178481784917850178511785217853178541785517856178571785817859178601786117862178631786417865178661786717868178691787017871178721787317874178751787617877178781787917880178811788217883178841788517886178871788817889178901789117892178931789417895178961789717898178991790017901179021790317904179051790617907179081790917910179111791217913179141791517916179171791817919179201792117922179231792417925179261792717928179291793017931179321793317934179351793617937179381793917940179411794217943179441794517946179471794817949179501795117952179531795417955179561795717958179591796017961179621796317964179651796617967179681796917970179711797217973179741797517976179771797817979179801798117982179831798417985179861798717988179891799017991179921799317994179951799617997179981799918000180011800218003180041800518006180071800818009180101801118012180131801418015180161801718018180191802018021180221802318024180251802618027180281802918030180311803218033180341803518036180371803818039180401804118042180431804418045180461804718048180491805018051180521805318054180551805618057180581805918060180611806218063180641806518066180671806818069180701807118072180731807418075180761807718078180791808018081180821808318084180851808618087180881808918090180911809218093180941809518096180971809818099181001810118102181031810418105181061810718108181091811018111181121811318114181151811618117181181811918120181211812218123181241812518126181271812818129181301813118132181331813418135181361813718138181391814018141181421814318144181451814618147181481814918150181511815218153181541815518156181571815818159181601816118162181631816418165181661816718168181691817018171181721817318174181751817618177181781817918180181811818218183181841818518186181871818818189181901819118192181931819418195181961819718198181991820018201182021820318204182051820618207182081820918210182111821218213182141821518216182171821818219182201822118222182231822418225182261822718228182291823018231182321823318234182351823618237182381823918240182411824218243182441824518246182471824818249182501825118252182531825418255182561825718258182591826018261182621826318264182651826618267182681826918270182711827218273182741827518276182771827818279182801828118282182831828418285182861828718288182891829018291182921829318294182951829618297182981829918300183011830218303183041830518306183071830818309183101831118312183131831418315183161831718318183191832018321183221832318324183251832618327183281832918330183311833218333183341833518336183371833818339183401834118342183431834418345183461834718348183491835018351183521835318354183551835618357183581835918360183611836218363183641836518366183671836818369183701837118372183731837418375183761837718378183791838018381183821838318384183851838618387183881838918390183911839218393183941839518396183971839818399184001840118402184031840418405184061840718408184091841018411184121841318414184151841618417184181841918420184211842218423184241842518426184271842818429184301843118432184331843418435184361843718438184391844018441184421844318444184451844618447184481844918450184511845218453184541845518456184571845818459184601846118462184631846418465184661846718468184691847018471184721847318474184751847618477184781847918480184811848218483184841848518486184871848818489184901849118492184931849418495184961849718498184991850018501185021850318504185051850618507185081850918510185111851218513185141851518516185171851818519185201852118522185231852418525185261852718528185291853018531185321853318534185351853618537185381853918540185411854218543185441854518546185471854818549185501855118552185531855418555185561855718558185591856018561185621856318564185651856618567185681856918570185711857218573185741857518576185771857818579185801858118582185831858418585185861858718588185891859018591185921859318594185951859618597185981859918600186011860218603186041860518606186071860818609186101861118612186131861418615186161861718618186191862018621186221862318624186251862618627186281862918630186311863218633186341863518636186371863818639186401864118642186431864418645186461864718648186491865018651186521865318654186551865618657186581865918660186611866218663186641866518666186671866818669186701867118672186731867418675186761867718678186791868018681186821868318684186851868618687186881868918690186911869218693186941869518696186971869818699187001870118702187031870418705187061870718708187091871018711187121871318714187151871618717187181871918720187211872218723187241872518726187271872818729187301873118732187331873418735187361873718738187391874018741187421874318744187451874618747187481874918750187511875218753187541875518756187571875818759187601876118762187631876418765187661876718768187691877018771187721877318774187751877618777187781877918780187811878218783187841878518786187871878818789187901879118792187931879418795187961879718798187991880018801188021880318804188051880618807188081880918810188111881218813188141881518816188171881818819188201882118822188231882418825188261882718828188291883018831188321883318834188351883618837188381883918840188411884218843188441884518846188471884818849188501885118852188531885418855188561885718858188591886018861188621886318864188651886618867188681886918870188711887218873188741887518876188771887818879188801888118882188831888418885188861888718888188891889018891188921889318894188951889618897188981889918900189011890218903189041890518906189071890818909189101891118912189131891418915189161891718918189191892018921189221892318924189251892618927189281892918930189311893218933189341893518936189371893818939189401894118942189431894418945189461894718948189491895018951189521895318954189551895618957189581895918960189611896218963189641896518966189671896818969189701897118972189731897418975189761897718978189791898018981189821898318984189851898618987189881898918990189911899218993189941899518996189971899818999190001900119002190031900419005190061900719008190091901019011190121901319014190151901619017190181901919020190211902219023190241902519026190271902819029190301903119032190331903419035190361903719038190391904019041190421904319044190451904619047190481904919050190511905219053190541905519056190571905819059190601906119062190631906419065190661906719068190691907019071190721907319074190751907619077190781907919080190811908219083190841908519086190871908819089190901909119092190931909419095190961909719098190991910019101191021910319104191051910619107191081910919110191111911219113191141911519116191171911819119191201912119122191231912419125191261912719128191291913019131191321913319134191351913619137191381913919140191411914219143191441914519146191471914819149191501915119152191531915419155191561915719158191591916019161191621916319164191651916619167191681916919170191711917219173191741917519176191771917819179191801918119182191831918419185191861918719188191891919019191191921919319194191951919619197191981919919200192011920219203192041920519206192071920819209192101921119212192131921419215192161921719218192191922019221192221922319224192251922619227192281922919230192311923219233192341923519236192371923819239192401924119242192431924419245192461924719248192491925019251192521925319254192551925619257192581925919260192611926219263192641926519266192671926819269192701927119272192731927419275192761927719278192791928019281192821928319284192851928619287192881928919290192911929219293192941929519296192971929819299193001930119302193031930419305193061930719308193091931019311193121931319314193151931619317193181931919320193211932219323193241932519326193271932819329193301933119332193331933419335193361933719338193391934019341193421934319344193451934619347193481934919350193511935219353193541935519356193571935819359193601936119362193631936419365193661936719368193691937019371193721937319374193751937619377193781937919380193811938219383193841938519386193871938819389193901939119392193931939419395193961939719398193991940019401194021940319404194051940619407194081940919410194111941219413194141941519416194171941819419194201942119422194231942419425194261942719428194291943019431194321943319434194351943619437194381943919440194411944219443194441944519446194471944819449194501945119452194531945419455194561945719458194591946019461194621946319464194651946619467194681946919470194711947219473194741947519476194771947819479194801948119482194831948419485194861948719488194891949019491194921949319494194951949619497194981949919500195011950219503195041950519506195071950819509195101951119512195131951419515195161951719518195191952019521195221952319524195251952619527195281952919530195311953219533195341953519536195371953819539195401954119542195431954419545195461954719548195491955019551195521955319554195551955619557195581955919560195611956219563195641956519566195671956819569195701957119572195731957419575195761957719578195791958019581195821958319584195851958619587195881958919590195911959219593195941959519596195971959819599196001960119602196031960419605196061960719608196091961019611196121961319614196151961619617196181961919620196211962219623196241962519626196271962819629196301963119632196331963419635196361963719638196391964019641196421964319644196451964619647196481964919650196511965219653196541965519656196571965819659196601966119662196631966419665196661966719668196691967019671196721967319674196751967619677196781967919680196811968219683196841968519686196871968819689196901969119692196931969419695196961969719698196991970019701197021970319704197051970619707197081970919710197111971219713197141971519716197171971819719197201972119722197231972419725197261972719728197291973019731197321973319734197351973619737197381973919740197411974219743197441974519746197471974819749197501975119752197531975419755197561975719758197591976019761197621976319764197651976619767197681976919770197711977219773197741977519776197771977819779197801978119782197831978419785197861978719788197891979019791197921979319794197951979619797197981979919800198011980219803198041980519806198071980819809198101981119812198131981419815198161981719818198191982019821198221982319824198251982619827198281982919830198311983219833198341983519836198371983819839198401984119842198431984419845198461984719848198491985019851198521985319854198551985619857198581985919860198611986219863198641986519866198671986819869198701987119872198731987419875198761987719878198791988019881198821988319884198851988619887198881988919890198911989219893198941989519896198971989819899199001990119902199031990419905199061990719908199091991019911199121991319914199151991619917199181991919920199211992219923199241992519926199271992819929199301993119932199331993419935199361993719938199391994019941199421994319944199451994619947199481994919950199511995219953199541995519956199571995819959199601996119962199631996419965199661996719968199691997019971199721997319974199751997619977199781997919980199811998219983199841998519986199871998819989199901999119992199931999419995199961999719998199992000020001200022000320004200052000620007200082000920010200112001220013200142001520016200172001820019200202002120022200232002420025200262002720028200292003020031200322003320034200352003620037200382003920040200412004220043200442004520046200472004820049200502005120052200532005420055200562005720058200592006020061200622006320064200652006620067200682006920070200712007220073200742007520076200772007820079200802008120082200832008420085200862008720088200892009020091200922009320094200952009620097200982009920100201012010220103201042010520106201072010820109201102011120112201132011420115201162011720118201192012020121201222012320124201252012620127201282012920130201312013220133201342013520136201372013820139201402014120142201432014420145201462014720148201492015020151201522015320154201552015620157201582015920160201612016220163201642016520166201672016820169201702017120172201732017420175201762017720178201792018020181201822018320184201852018620187201882018920190201912019220193201942019520196201972019820199202002020120202202032020420205202062020720208202092021020211202122021320214202152021620217202182021920220202212022220223202242022520226202272022820229202302023120232202332023420235202362023720238202392024020241202422024320244202452024620247202482024920250202512025220253202542025520256202572025820259202602026120262202632026420265202662026720268202692027020271202722027320274202752027620277202782027920280202812028220283202842028520286202872028820289202902029120292202932029420295202962029720298202992030020301203022030320304203052030620307203082030920310203112031220313203142031520316203172031820319203202032120322203232032420325203262032720328203292033020331203322033320334203352033620337203382033920340203412034220343203442034520346203472034820349203502035120352203532035420355203562035720358203592036020361203622036320364203652036620367203682036920370203712037220373203742037520376203772037820379203802038120382203832038420385203862038720388203892039020391203922039320394203952039620397203982039920400204012040220403204042040520406204072040820409204102041120412204132041420415204162041720418204192042020421204222042320424204252042620427204282042920430204312043220433204342043520436204372043820439204402044120442204432044420445204462044720448204492045020451204522045320454204552045620457204582045920460204612046220463204642046520466204672046820469204702047120472204732047420475204762047720478204792048020481204822048320484204852048620487204882048920490204912049220493204942049520496204972049820499205002050120502205032050420505205062050720508205092051020511205122051320514205152051620517205182051920520205212052220523205242052520526205272052820529205302053120532205332053420535205362053720538205392054020541205422054320544205452054620547205482054920550205512055220553205542055520556205572055820559205602056120562205632056420565205662056720568205692057020571205722057320574205752057620577205782057920580205812058220583205842058520586205872058820589205902059120592205932059420595205962059720598205992060020601206022060320604206052060620607206082060920610206112061220613206142061520616206172061820619206202062120622206232062420625206262062720628206292063020631206322063320634206352063620637206382063920640206412064220643206442064520646206472064820649206502065120652206532065420655206562065720658206592066020661206622066320664206652066620667206682066920670206712067220673206742067520676206772067820679206802068120682206832068420685206862068720688206892069020691206922069320694206952069620697206982069920700207012070220703207042070520706207072070820709207102071120712207132071420715207162071720718207192072020721207222072320724207252072620727207282072920730207312073220733207342073520736207372073820739207402074120742207432074420745207462074720748207492075020751207522075320754207552075620757207582075920760207612076220763207642076520766207672076820769207702077120772207732077420775207762077720778207792078020781207822078320784207852078620787207882078920790207912079220793207942079520796207972079820799208002080120802208032080420805208062080720808208092081020811208122081320814208152081620817208182081920820208212082220823208242082520826208272082820829208302083120832208332083420835208362083720838208392084020841208422084320844208452084620847208482084920850208512085220853208542085520856208572085820859208602086120862208632086420865208662086720868208692087020871208722087320874208752087620877208782087920880208812088220883208842088520886208872088820889208902089120892208932089420895208962089720898208992090020901209022090320904209052090620907209082090920910209112091220913209142091520916209172091820919209202092120922209232092420925209262092720928209292093020931209322093320934209352093620937209382093920940209412094220943209442094520946209472094820949209502095120952209532095420955209562095720958209592096020961209622096320964209652096620967209682096920970209712097220973209742097520976209772097820979209802098120982209832098420985209862098720988209892099020991209922099320994209952099620997209982099921000210012100221003210042100521006210072100821009210102101121012210132101421015210162101721018210192102021021210222102321024210252102621027210282102921030210312103221033210342103521036210372103821039210402104121042210432104421045210462104721048210492105021051210522105321054210552105621057210582105921060210612106221063210642106521066210672106821069210702107121072210732107421075210762107721078210792108021081210822108321084210852108621087210882108921090210912109221093210942109521096210972109821099211002110121102211032110421105211062110721108211092111021111211122111321114211152111621117211182111921120211212112221123211242112521126211272112821129211302113121132211332113421135211362113721138211392114021141211422114321144211452114621147211482114921150211512115221153211542115521156211572115821159211602116121162211632116421165211662116721168211692117021171211722117321174211752117621177211782117921180211812118221183211842118521186211872118821189211902119121192211932119421195211962119721198211992120021201212022120321204212052120621207212082120921210212112121221213212142121521216212172121821219212202122121222212232122421225212262122721228212292123021231212322123321234212352123621237212382123921240212412124221243212442124521246212472124821249212502125121252212532125421255212562125721258212592126021261212622126321264212652126621267212682126921270212712127221273212742127521276212772127821279212802128121282212832128421285212862128721288212892129021291212922129321294212952129621297212982129921300213012130221303213042130521306213072130821309213102131121312213132131421315213162131721318213192132021321213222132321324213252132621327213282132921330213312133221333213342133521336213372133821339213402134121342213432134421345213462134721348213492135021351213522135321354213552135621357213582135921360213612136221363213642136521366213672136821369213702137121372213732137421375213762137721378213792138021381213822138321384213852138621387213882138921390213912139221393213942139521396213972139821399214002140121402214032140421405214062140721408214092141021411214122141321414214152141621417214182141921420214212142221423214242142521426214272142821429214302143121432214332143421435214362143721438214392144021441214422144321444214452144621447214482144921450214512145221453214542145521456214572145821459214602146121462214632146421465214662146721468214692147021471214722147321474214752147621477214782147921480214812148221483214842148521486214872148821489214902149121492214932149421495214962149721498214992150021501215022150321504215052150621507215082150921510215112151221513215142151521516215172151821519215202152121522215232152421525215262152721528215292153021531215322153321534215352153621537215382153921540215412154221543215442154521546215472154821549215502155121552215532155421555215562155721558215592156021561215622156321564215652156621567215682156921570215712157221573215742157521576215772157821579215802158121582215832158421585215862158721588215892159021591215922159321594215952159621597215982159921600216012160221603216042160521606216072160821609216102161121612216132161421615216162161721618216192162021621216222162321624216252162621627216282162921630216312163221633216342163521636216372163821639216402164121642216432164421645216462164721648216492165021651216522165321654216552165621657216582165921660216612166221663216642166521666216672166821669216702167121672216732167421675216762167721678216792168021681216822168321684216852168621687216882168921690216912169221693216942169521696216972169821699217002170121702217032170421705217062170721708217092171021711217122171321714217152171621717217182171921720217212172221723217242172521726217272172821729217302173121732217332173421735217362173721738217392174021741217422174321744217452174621747217482174921750217512175221753217542175521756217572175821759217602176121762217632176421765217662176721768217692177021771217722177321774217752177621777217782177921780217812178221783217842178521786217872178821789217902179121792217932179421795217962179721798217992180021801218022180321804218052180621807218082180921810218112181221813218142181521816218172181821819218202182121822218232182421825218262182721828218292183021831218322183321834218352183621837218382183921840218412184221843218442184521846218472184821849218502185121852218532185421855218562185721858218592186021861218622186321864218652186621867218682186921870218712187221873218742187521876218772187821879218802188121882218832188421885218862188721888218892189021891218922189321894218952189621897218982189921900219012190221903219042190521906219072190821909219102191121912219132191421915219162191721918219192192021921219222192321924219252192621927219282192921930219312193221933219342193521936219372193821939219402194121942219432194421945219462194721948219492195021951219522195321954219552195621957219582195921960219612196221963219642196521966219672196821969219702197121972219732197421975219762197721978219792198021981219822198321984219852198621987219882198921990219912199221993219942199521996219972199821999220002200122002220032200422005220062200722008220092201022011220122201322014220152201622017220182201922020220212202222023220242202522026220272202822029220302203122032220332203422035220362203722038220392204022041220422204322044220452204622047220482204922050220512205222053220542205522056220572205822059220602206122062220632206422065220662206722068220692207022071220722207322074220752207622077220782207922080220812208222083220842208522086220872208822089220902209122092220932209422095220962209722098220992210022101221022210322104221052210622107221082210922110221112211222113221142211522116221172211822119221202212122122221232212422125221262212722128221292213022131221322213322134221352213622137221382213922140221412214222143221442214522146221472214822149221502215122152221532215422155221562215722158221592216022161221622216322164221652216622167221682216922170221712217222173221742217522176221772217822179221802218122182221832218422185221862218722188221892219022191221922219322194221952219622197221982219922200222012220222203222042220522206222072220822209222102221122212222132221422215222162221722218222192222022221222222222322224222252222622227222282222922230222312223222233222342223522236222372223822239222402224122242222432224422245222462224722248222492225022251222522225322254222552225622257222582225922260222612226222263222642226522266222672226822269222702227122272222732227422275222762227722278222792228022281222822228322284222852228622287222882228922290222912229222293222942229522296222972229822299223002230122302223032230422305223062230722308223092231022311223122231322314223152231622317223182231922320223212232222323223242232522326223272232822329223302233122332223332233422335223362233722338223392234022341223422234322344223452234622347223482234922350223512235222353223542235522356223572235822359223602236122362223632236422365223662236722368223692237022371223722237322374223752237622377223782237922380223812238222383223842238522386223872238822389223902239122392223932239422395223962239722398223992240022401224022240322404224052240622407224082240922410224112241222413224142241522416224172241822419224202242122422224232242422425224262242722428224292243022431224322243322434224352243622437224382243922440224412244222443224442244522446224472244822449224502245122452224532245422455224562245722458224592246022461224622246322464224652246622467224682246922470224712247222473224742247522476224772247822479224802248122482224832248422485224862248722488224892249022491224922249322494224952249622497224982249922500225012250222503225042250522506225072250822509225102251122512225132251422515225162251722518225192252022521225222252322524225252252622527225282252922530225312253222533225342253522536225372253822539225402254122542225432254422545225462254722548225492255022551225522255322554225552255622557225582255922560225612256222563225642256522566225672256822569225702257122572225732257422575225762257722578225792258022581225822258322584225852258622587225882258922590225912259222593225942259522596225972259822599226002260122602226032260422605226062260722608226092261022611226122261322614226152261622617226182261922620226212262222623226242262522626226272262822629226302263122632226332263422635226362263722638226392264022641226422264322644226452264622647226482264922650226512265222653226542265522656226572265822659226602266122662226632266422665226662266722668226692267022671226722267322674226752267622677226782267922680226812268222683226842268522686226872268822689226902269122692226932269422695226962269722698226992270022701227022270322704227052270622707227082270922710227112271222713227142271522716227172271822719227202272122722227232272422725227262272722728227292273022731227322273322734227352273622737227382273922740227412274222743227442274522746227472274822749227502275122752227532275422755227562275722758227592276022761227622276322764227652276622767227682276922770227712277222773227742277522776227772277822779227802278122782227832278422785227862278722788227892279022791227922279322794227952279622797227982279922800228012280222803228042280522806228072280822809228102281122812228132281422815228162281722818228192282022821228222282322824228252282622827228282282922830228312283222833228342283522836228372283822839228402284122842228432284422845228462284722848228492285022851228522285322854228552285622857228582285922860228612286222863228642286522866228672286822869228702287122872228732287422875228762287722878228792288022881228822288322884228852288622887228882288922890228912289222893228942289522896228972289822899229002290122902229032290422905229062290722908229092291022911229122291322914229152291622917229182291922920229212292222923229242292522926229272292822929229302293122932229332293422935229362293722938229392294022941229422294322944229452294622947229482294922950229512295222953229542295522956229572295822959229602296122962229632296422965229662296722968229692297022971229722297322974229752297622977229782297922980229812298222983229842298522986229872298822989229902299122992229932299422995229962299722998229992300023001230022300323004230052300623007230082300923010230112301223013230142301523016230172301823019230202302123022230232302423025230262302723028230292303023031230322303323034230352303623037230382303923040230412304223043230442304523046230472304823049230502305123052230532305423055230562305723058230592306023061230622306323064230652306623067230682306923070230712307223073230742307523076230772307823079230802308123082230832308423085230862308723088230892309023091230922309323094230952309623097230982309923100231012310223103231042310523106231072310823109231102311123112231132311423115231162311723118231192312023121231222312323124231252312623127231282312923130231312313223133231342313523136231372313823139231402314123142231432314423145231462314723148231492315023151231522315323154231552315623157231582315923160231612316223163231642316523166231672316823169231702317123172231732317423175231762317723178231792318023181231822318323184231852318623187231882318923190231912319223193231942319523196231972319823199232002320123202232032320423205232062320723208232092321023211232122321323214232152321623217232182321923220232212322223223232242322523226232272322823229232302323123232232332323423235232362323723238232392324023241232422324323244232452324623247232482324923250232512325223253232542325523256232572325823259232602326123262232632326423265232662326723268232692327023271232722327323274232752327623277232782327923280232812328223283232842328523286232872328823289232902329123292232932329423295232962329723298232992330023301233022330323304233052330623307233082330923310233112331223313233142331523316233172331823319233202332123322233232332423325233262332723328233292333023331233322333323334233352333623337233382333923340233412334223343233442334523346233472334823349233502335123352233532335423355233562335723358233592336023361233622336323364233652336623367233682336923370233712337223373233742337523376233772337823379233802338123382233832338423385233862338723388233892339023391233922339323394233952339623397233982339923400234012340223403234042340523406234072340823409234102341123412234132341423415234162341723418234192342023421234222342323424234252342623427234282342923430234312343223433234342343523436234372343823439234402344123442234432344423445234462344723448234492345023451234522345323454234552345623457234582345923460234612346223463234642346523466234672346823469234702347123472234732347423475234762347723478234792348023481234822348323484234852348623487234882348923490234912349223493234942349523496234972349823499235002350123502235032350423505235062350723508235092351023511235122351323514235152351623517235182351923520235212352223523235242352523526235272352823529235302353123532235332353423535235362353723538235392354023541235422354323544235452354623547235482354923550235512355223553235542355523556235572355823559235602356123562235632356423565235662356723568235692357023571235722357323574235752357623577235782357923580235812358223583235842358523586235872358823589235902359123592235932359423595235962359723598235992360023601236022360323604236052360623607236082360923610236112361223613236142361523616236172361823619236202362123622236232362423625236262362723628236292363023631236322363323634236352363623637236382363923640236412364223643236442364523646236472364823649236502365123652236532365423655236562365723658236592366023661236622366323664236652366623667236682366923670236712367223673236742367523676236772367823679236802368123682236832368423685236862368723688236892369023691236922369323694236952369623697236982369923700237012370223703237042370523706237072370823709237102371123712237132371423715237162371723718237192372023721237222372323724237252372623727237282372923730237312373223733237342373523736237372373823739237402374123742237432374423745237462374723748237492375023751237522375323754237552375623757237582375923760237612376223763237642376523766237672376823769237702377123772237732377423775237762377723778237792378023781237822378323784237852378623787237882378923790237912379223793237942379523796237972379823799238002380123802238032380423805238062380723808238092381023811238122381323814238152381623817238182381923820238212382223823238242382523826238272382823829238302383123832238332383423835238362383723838238392384023841238422384323844238452384623847238482384923850238512385223853238542385523856238572385823859238602386123862238632386423865238662386723868238692387023871238722387323874238752387623877238782387923880238812388223883238842388523886238872388823889238902389123892238932389423895238962389723898238992390023901239022390323904239052390623907239082390923910239112391223913239142391523916239172391823919239202392123922239232392423925239262392723928239292393023931239322393323934239352393623937239382393923940239412394223943239442394523946239472394823949239502395123952239532395423955239562395723958239592396023961239622396323964239652396623967239682396923970239712397223973239742397523976239772397823979239802398123982239832398423985239862398723988239892399023991239922399323994239952399623997239982399924000240012400224003240042400524006240072400824009240102401124012240132401424015240162401724018240192402024021240222402324024240252402624027240282402924030240312403224033240342403524036240372403824039240402404124042240432404424045240462404724048240492405024051240522405324054240552405624057240582405924060240612406224063240642406524066240672406824069240702407124072240732407424075240762407724078240792408024081240822408324084240852408624087240882408924090240912409224093240942409524096240972409824099241002410124102241032410424105241062410724108241092411024111241122411324114241152411624117241182411924120241212412224123241242412524126241272412824129241302413124132241332413424135241362413724138241392414024141241422414324144241452414624147241482414924150241512415224153241542415524156241572415824159241602416124162241632416424165241662416724168241692417024171241722417324174241752417624177241782417924180241812418224183241842418524186241872418824189241902419124192241932419424195241962419724198241992420024201242022420324204242052420624207242082420924210242112421224213242142421524216242172421824219242202422124222242232422424225242262422724228242292423024231242322423324234242352423624237242382423924240242412424224243242442424524246242472424824249242502425124252242532425424255242562425724258242592426024261242622426324264242652426624267242682426924270242712427224273242742427524276242772427824279242802428124282242832428424285242862428724288242892429024291242922429324294242952429624297242982429924300243012430224303243042430524306243072430824309243102431124312243132431424315243162431724318243192432024321243222432324324243252432624327243282432924330243312433224333243342433524336243372433824339243402434124342243432434424345243462434724348243492435024351243522435324354243552435624357243582435924360243612436224363243642436524366243672436824369243702437124372243732437424375243762437724378243792438024381243822438324384243852438624387243882438924390243912439224393243942439524396243972439824399244002440124402244032440424405244062440724408244092441024411244122441324414244152441624417244182441924420244212442224423244242442524426244272442824429244302443124432244332443424435244362443724438244392444024441244422444324444244452444624447244482444924450244512445224453244542445524456244572445824459244602446124462244632446424465244662446724468244692447024471244722447324474244752447624477244782447924480244812448224483244842448524486244872448824489244902449124492244932449424495244962449724498244992450024501245022450324504245052450624507245082450924510245112451224513245142451524516245172451824519245202452124522245232452424525245262452724528245292453024531245322453324534245352453624537245382453924540245412454224543245442454524546245472454824549245502455124552245532455424555245562455724558245592456024561245622456324564245652456624567245682456924570245712457224573245742457524576245772457824579245802458124582245832458424585245862458724588245892459024591245922459324594245952459624597245982459924600246012460224603246042460524606246072460824609246102461124612246132461424615246162461724618246192462024621246222462324624246252462624627246282462924630246312463224633246342463524636246372463824639246402464124642246432464424645246462464724648246492465024651246522465324654246552465624657246582465924660246612466224663246642466524666246672466824669246702467124672246732467424675246762467724678246792468024681246822468324684246852468624687246882468924690246912469224693246942469524696246972469824699247002470124702247032470424705247062470724708247092471024711247122471324714247152471624717247182471924720247212472224723247242472524726247272472824729247302473124732247332473424735247362473724738247392474024741247422474324744247452474624747247482474924750247512475224753247542475524756247572475824759247602476124762247632476424765247662476724768247692477024771247722477324774247752477624777247782477924780247812478224783247842478524786247872478824789247902479124792247932479424795247962479724798247992480024801248022480324804248052480624807248082480924810248112481224813248142481524816248172481824819248202482124822248232482424825248262482724828248292483024831248322483324834248352483624837248382483924840248412484224843248442484524846248472484824849248502485124852248532485424855248562485724858248592486024861248622486324864248652486624867248682486924870248712487224873248742487524876248772487824879248802488124882248832488424885248862488724888248892489024891248922489324894248952489624897248982489924900249012490224903249042490524906249072490824909249102491124912249132491424915249162491724918249192492024921249222492324924249252492624927249282492924930249312493224933249342493524936249372493824939249402494124942249432494424945249462494724948249492495024951249522495324954249552495624957249582495924960249612496224963249642496524966249672496824969249702497124972249732497424975249762497724978249792498024981249822498324984249852498624987249882498924990249912499224993249942499524996249972499824999250002500125002250032500425005250062500725008250092501025011250122501325014250152501625017250182501925020250212502225023250242502525026250272502825029250302503125032250332503425035250362503725038250392504025041250422504325044250452504625047250482504925050250512505225053250542505525056250572505825059250602506125062250632506425065250662506725068250692507025071250722507325074250752507625077250782507925080250812508225083250842508525086250872508825089250902509125092250932509425095250962509725098250992510025101251022510325104251052510625107251082510925110251112511225113251142511525116251172511825119251202512125122251232512425125251262512725128251292513025131251322513325134251352513625137251382513925140251412514225143251442514525146251472514825149251502515125152251532515425155251562515725158251592516025161251622516325164251652516625167251682516925170251712517225173251742517525176251772517825179251802518125182251832518425185251862518725188251892519025191251922519325194251952519625197251982519925200252012520225203252042520525206252072520825209252102521125212252132521425215252162521725218252192522025221252222522325224252252522625227252282522925230252312523225233252342523525236252372523825239252402524125242252432524425245252462524725248252492525025251252522525325254252552525625257252582525925260252612526225263252642526525266252672526825269252702527125272252732527425275252762527725278252792528025281252822528325284252852528625287252882528925290252912529225293252942529525296252972529825299253002530125302253032530425305253062530725308253092531025311253122531325314253152531625317253182531925320253212532225323253242532525326253272532825329253302533125332253332533425335253362533725338253392534025341253422534325344253452534625347253482534925350253512535225353253542535525356253572535825359253602536125362253632536425365253662536725368253692537025371253722537325374253752537625377253782537925380253812538225383253842538525386253872538825389253902539125392253932539425395253962539725398253992540025401254022540325404254052540625407254082540925410254112541225413254142541525416254172541825419254202542125422254232542425425254262542725428254292543025431254322543325434254352543625437254382543925440254412544225443254442544525446254472544825449254502545125452254532545425455254562545725458254592546025461254622546325464254652546625467254682546925470254712547225473254742547525476254772547825479254802548125482254832548425485254862548725488254892549025491254922549325494254952549625497254982549925500255012550225503255042550525506255072550825509255102551125512255132551425515255162551725518255192552025521255222552325524255252552625527255282552925530255312553225533255342553525536255372553825539255402554125542255432554425545255462554725548255492555025551255522555325554255552555625557255582555925560255612556225563255642556525566255672556825569255702557125572255732557425575255762557725578255792558025581255822558325584255852558625587255882558925590255912559225593255942559525596255972559825599256002560125602256032560425605256062560725608256092561025611256122561325614256152561625617256182561925620256212562225623256242562525626256272562825629256302563125632256332563425635256362563725638256392564025641256422564325644256452564625647256482564925650256512565225653256542565525656256572565825659256602566125662256632566425665256662566725668256692567025671256722567325674256752567625677256782567925680256812568225683256842568525686256872568825689256902569125692256932569425695256962569725698256992570025701257022570325704257052570625707257082570925710257112571225713257142571525716257172571825719257202572125722257232572425725257262572725728257292573025731257322573325734257352573625737257382573925740257412574225743257442574525746257472574825749257502575125752257532575425755257562575725758257592576025761257622576325764257652576625767257682576925770257712577225773257742577525776257772577825779257802578125782257832578425785257862578725788257892579025791257922579325794257952579625797257982579925800258012580225803258042580525806258072580825809258102581125812258132581425815258162581725818258192582025821258222582325824258252582625827258282582925830258312583225833258342583525836258372583825839258402584125842258432584425845258462584725848258492585025851258522585325854258552585625857258582585925860258612586225863258642586525866258672586825869258702587125872258732587425875258762587725878258792588025881258822588325884258852588625887258882588925890258912589225893258942589525896258972589825899259002590125902259032590425905259062590725908259092591025911259122591325914259152591625917259182591925920259212592225923259242592525926259272592825929259302593125932259332593425935259362593725938259392594025941259422594325944259452594625947259482594925950259512595225953259542595525956259572595825959259602596125962259632596425965259662596725968259692597025971259722597325974259752597625977259782597925980259812598225983259842598525986259872598825989259902599125992259932599425995259962599725998259992600026001260022600326004260052600626007260082600926010260112601226013260142601526016260172601826019260202602126022260232602426025260262602726028260292603026031260322603326034260352603626037260382603926040260412604226043260442604526046260472604826049260502605126052260532605426055260562605726058260592606026061260622606326064260652606626067260682606926070260712607226073260742607526076260772607826079260802608126082260832608426085260862608726088260892609026091260922609326094260952609626097260982609926100261012610226103261042610526106261072610826109261102611126112261132611426115261162611726118261192612026121261222612326124261252612626127261282612926130261312613226133261342613526136261372613826139261402614126142261432614426145261462614726148261492615026151261522615326154261552615626157261582615926160261612616226163261642616526166261672616826169261702617126172261732617426175261762617726178261792618026181261822618326184261852618626187261882618926190261912619226193261942619526196261972619826199262002620126202262032620426205262062620726208262092621026211262122621326214262152621626217262182621926220262212622226223262242622526226262272622826229262302623126232262332623426235262362623726238262392624026241262422624326244262452624626247262482624926250262512625226253262542625526256262572625826259262602626126262262632626426265262662626726268262692627026271262722627326274262752627626277262782627926280262812628226283262842628526286262872628826289262902629126292262932629426295262962629726298262992630026301263022630326304263052630626307263082630926310263112631226313263142631526316263172631826319263202632126322263232632426325263262632726328263292633026331263322633326334263352633626337263382633926340263412634226343263442634526346263472634826349263502635126352263532635426355263562635726358263592636026361263622636326364263652636626367263682636926370263712637226373263742637526376263772637826379263802638126382263832638426385263862638726388263892639026391263922639326394263952639626397263982639926400264012640226403264042640526406264072640826409264102641126412264132641426415264162641726418264192642026421264222642326424264252642626427264282642926430264312643226433264342643526436264372643826439264402644126442264432644426445264462644726448264492645026451264522645326454264552645626457264582645926460264612646226463264642646526466264672646826469264702647126472264732647426475264762647726478264792648026481264822648326484264852648626487264882648926490264912649226493264942649526496264972649826499265002650126502265032650426505265062650726508265092651026511265122651326514265152651626517265182651926520265212652226523265242652526526265272652826529265302653126532265332653426535265362653726538265392654026541265422654326544265452654626547265482654926550265512655226553265542655526556265572655826559265602656126562265632656426565265662656726568265692657026571265722657326574265752657626577265782657926580265812658226583265842658526586265872658826589265902659126592265932659426595265962659726598265992660026601266022660326604266052660626607266082660926610266112661226613266142661526616266172661826619266202662126622266232662426625266262662726628266292663026631266322663326634266352663626637266382663926640266412664226643266442664526646266472664826649266502665126652266532665426655266562665726658266592666026661266622666326664266652666626667266682666926670266712667226673266742667526676266772667826679266802668126682266832668426685266862668726688266892669026691266922669326694266952669626697266982669926700267012670226703267042670526706267072670826709267102671126712267132671426715267162671726718267192672026721267222672326724267252672626727267282672926730267312673226733267342673526736267372673826739267402674126742267432674426745267462674726748267492675026751267522675326754267552675626757267582675926760267612676226763267642676526766267672676826769267702677126772267732677426775267762677726778267792678026781267822678326784267852678626787267882678926790267912679226793267942679526796267972679826799268002680126802268032680426805268062680726808268092681026811268122681326814268152681626817268182681926820268212682226823268242682526826268272682826829268302683126832268332683426835268362683726838268392684026841268422684326844268452684626847268482684926850268512685226853268542685526856268572685826859268602686126862268632686426865268662686726868268692687026871268722687326874268752687626877268782687926880268812688226883268842688526886268872688826889268902689126892268932689426895268962689726898268992690026901269022690326904269052690626907269082690926910269112691226913269142691526916269172691826919269202692126922269232692426925269262692726928269292693026931269322693326934269352693626937269382693926940269412694226943269442694526946269472694826949269502695126952269532695426955269562695726958269592696026961269622696326964269652696626967269682696926970269712697226973269742697526976269772697826979269802698126982269832698426985269862698726988269892699026991269922699326994269952699626997269982699927000270012700227003270042700527006270072700827009270102701127012270132701427015270162701727018270192702027021270222702327024270252702627027270282702927030270312703227033270342703527036270372703827039270402704127042270432704427045270462704727048270492705027051270522705327054270552705627057270582705927060270612706227063270642706527066270672706827069270702707127072270732707427075270762707727078270792708027081270822708327084270852708627087270882708927090270912709227093270942709527096270972709827099271002710127102271032710427105271062710727108271092711027111271122711327114271152711627117271182711927120271212712227123271242712527126271272712827129271302713127132271332713427135271362713727138271392714027141271422714327144271452714627147271482714927150271512715227153271542715527156271572715827159271602716127162271632716427165271662716727168271692717027171271722717327174271752717627177271782717927180271812718227183271842718527186271872718827189271902719127192271932719427195271962719727198271992720027201272022720327204272052720627207272082720927210272112721227213272142721527216272172721827219272202722127222272232722427225272262722727228272292723027231272322723327234272352723627237272382723927240272412724227243272442724527246272472724827249272502725127252272532725427255272562725727258272592726027261272622726327264272652726627267272682726927270272712727227273272742727527276272772727827279272802728127282272832728427285272862728727288272892729027291272922729327294272952729627297272982729927300273012730227303273042730527306273072730827309273102731127312273132731427315273162731727318273192732027321273222732327324273252732627327273282732927330273312733227333273342733527336273372733827339273402734127342273432734427345273462734727348273492735027351273522735327354273552735627357273582735927360273612736227363273642736527366273672736827369273702737127372273732737427375273762737727378273792738027381273822738327384273852738627387273882738927390273912739227393273942739527396273972739827399274002740127402274032740427405274062740727408274092741027411274122741327414274152741627417274182741927420274212742227423274242742527426274272742827429274302743127432274332743427435274362743727438274392744027441274422744327444274452744627447274482744927450274512745227453274542745527456274572745827459274602746127462274632746427465274662746727468274692747027471274722747327474274752747627477274782747927480274812748227483274842748527486274872748827489274902749127492274932749427495274962749727498274992750027501275022750327504275052750627507275082750927510275112751227513275142751527516275172751827519275202752127522275232752427525275262752727528275292753027531275322753327534275352753627537275382753927540275412754227543275442754527546275472754827549275502755127552275532755427555275562755727558275592756027561275622756327564275652756627567275682756927570275712757227573275742757527576275772757827579275802758127582275832758427585275862758727588275892759027591275922759327594275952759627597275982759927600276012760227603276042760527606276072760827609276102761127612276132761427615276162761727618276192762027621276222762327624276252762627627276282762927630276312763227633276342763527636276372763827639276402764127642276432764427645276462764727648276492765027651276522765327654276552765627657276582765927660276612766227663276642766527666276672766827669276702767127672276732767427675276762767727678276792768027681276822768327684276852768627687276882768927690276912769227693276942769527696276972769827699277002770127702277032770427705277062770727708277092771027711277122771327714277152771627717277182771927720277212772227723277242772527726277272772827729277302773127732277332773427735277362773727738277392774027741277422774327744277452774627747277482774927750277512775227753277542775527756277572775827759277602776127762277632776427765277662776727768277692777027771277722777327774277752777627777277782777927780277812778227783277842778527786277872778827789277902779127792277932779427795277962779727798277992780027801278022780327804278052780627807278082780927810278112781227813278142781527816278172781827819278202782127822278232782427825278262782727828278292783027831278322783327834278352783627837278382783927840278412784227843278442784527846278472784827849278502785127852278532785427855278562785727858278592786027861278622786327864278652786627867278682786927870278712787227873278742787527876278772787827879278802788127882278832788427885278862788727888278892789027891278922789327894278952789627897278982789927900279012790227903279042790527906279072790827909279102791127912279132791427915279162791727918279192792027921279222792327924279252792627927279282792927930279312793227933279342793527936279372793827939279402794127942279432794427945279462794727948279492795027951279522795327954279552795627957279582795927960279612796227963279642796527966279672796827969279702797127972279732797427975279762797727978279792798027981279822798327984279852798627987279882798927990279912799227993279942799527996279972799827999280002800128002280032800428005280062800728008280092801028011280122801328014280152801628017280182801928020280212802228023280242802528026280272802828029280302803128032280332803428035280362803728038280392804028041280422804328044280452804628047280482804928050280512805228053280542805528056280572805828059280602806128062280632806428065280662806728068280692807028071280722807328074280752807628077280782807928080280812808228083280842808528086280872808828089280902809128092280932809428095280962809728098280992810028101281022810328104281052810628107281082810928110281112811228113281142811528116281172811828119281202812128122281232812428125281262812728128281292813028131281322813328134281352813628137281382813928140281412814228143281442814528146281472814828149281502815128152281532815428155281562815728158281592816028161281622816328164281652816628167281682816928170281712817228173281742817528176281772817828179281802818128182281832818428185281862818728188281892819028191281922819328194281952819628197281982819928200282012820228203282042820528206282072820828209282102821128212282132821428215282162821728218282192822028221282222822328224282252822628227282282822928230282312823228233282342823528236282372823828239282402824128242282432824428245282462824728248282492825028251282522825328254282552825628257282582825928260282612826228263282642826528266282672826828269282702827128272282732827428275282762827728278282792828028281282822828328284282852828628287282882828928290282912829228293282942829528296282972829828299283002830128302283032830428305283062830728308283092831028311283122831328314283152831628317283182831928320283212832228323283242832528326283272832828329283302833128332283332833428335283362833728338283392834028341283422834328344283452834628347283482834928350283512835228353283542835528356283572835828359283602836128362283632836428365283662836728368283692837028371283722837328374283752837628377283782837928380283812838228383283842838528386283872838828389283902839128392283932839428395283962839728398283992840028401284022840328404284052840628407284082840928410284112841228413284142841528416284172841828419284202842128422284232842428425284262842728428284292843028431284322843328434284352843628437284382843928440284412844228443284442844528446284472844828449284502845128452284532845428455284562845728458284592846028461284622846328464284652846628467284682846928470284712847228473284742847528476284772847828479284802848128482284832848428485284862848728488284892849028491284922849328494284952849628497284982849928500285012850228503285042850528506285072850828509285102851128512285132851428515285162851728518285192852028521285222852328524285252852628527285282852928530285312853228533285342853528536285372853828539285402854128542285432854428545285462854728548285492855028551285522855328554285552855628557285582855928560285612856228563285642856528566285672856828569285702857128572285732857428575285762857728578285792858028581285822858328584285852858628587285882858928590285912859228593285942859528596285972859828599286002860128602286032860428605286062860728608286092861028611286122861328614286152861628617286182861928620286212862228623286242862528626286272862828629286302863128632286332863428635286362863728638286392864028641286422864328644286452864628647286482864928650286512865228653286542865528656286572865828659286602866128662286632866428665286662866728668286692867028671286722867328674286752867628677286782867928680286812868228683286842868528686286872868828689286902869128692286932869428695286962869728698286992870028701287022870328704287052870628707287082870928710287112871228713287142871528716287172871828719287202872128722287232872428725287262872728728287292873028731287322873328734287352873628737287382873928740287412874228743287442874528746287472874828749287502875128752287532875428755287562875728758287592876028761287622876328764287652876628767287682876928770287712877228773287742877528776287772877828779287802878128782287832878428785287862878728788287892879028791287922879328794287952879628797287982879928800288012880228803288042880528806288072880828809288102881128812288132881428815288162881728818288192882028821288222882328824288252882628827288282882928830288312883228833288342883528836288372883828839288402884128842288432884428845288462884728848288492885028851288522885328854288552885628857288582885928860288612886228863288642886528866288672886828869288702887128872288732887428875288762887728878288792888028881288822888328884288852888628887288882888928890288912889228893288942889528896288972889828899289002890128902289032890428905289062890728908289092891028911289122891328914289152891628917289182891928920289212892228923289242892528926289272892828929289302893128932289332893428935289362893728938289392894028941289422894328944289452894628947289482894928950289512895228953289542895528956289572895828959289602896128962289632896428965289662896728968289692897028971289722897328974289752897628977289782897928980289812898228983289842898528986289872898828989289902899128992289932899428995289962899728998289992900029001290022900329004290052900629007290082900929010290112901229013290142901529016290172901829019290202902129022290232902429025290262902729028290292903029031290322903329034290352903629037290382903929040290412904229043290442904529046290472904829049290502905129052290532905429055290562905729058290592906029061290622906329064290652906629067290682906929070290712907229073290742907529076290772907829079290802908129082290832908429085290862908729088290892909029091290922909329094290952909629097290982909929100291012910229103291042910529106291072910829109291102911129112291132911429115291162911729118291192912029121291222912329124291252912629127291282912929130291312913229133291342913529136291372913829139291402914129142291432914429145291462914729148291492915029151291522915329154291552915629157291582915929160291612916229163291642916529166291672916829169291702917129172291732917429175291762917729178291792918029181291822918329184291852918629187291882918929190291912919229193291942919529196291972919829199292002920129202292032920429205292062920729208292092921029211292122921329214292152921629217292182921929220292212922229223292242922529226292272922829229292302923129232292332923429235292362923729238292392924029241292422924329244292452924629247292482924929250292512925229253292542925529256292572925829259292602926129262292632926429265292662926729268292692927029271292722927329274292752927629277292782927929280292812928229283292842928529286292872928829289292902929129292292932929429295292962929729298292992930029301293022930329304293052930629307293082930929310293112931229313293142931529316293172931829319293202932129322293232932429325293262932729328293292933029331293322933329334293352933629337293382933929340293412934229343293442934529346293472934829349293502935129352293532935429355293562935729358293592936029361293622936329364293652936629367293682936929370293712937229373293742937529376293772937829379293802938129382293832938429385293862938729388293892939029391293922939329394293952939629397293982939929400294012940229403294042940529406294072940829409294102941129412294132941429415294162941729418294192942029421294222942329424294252942629427294282942929430294312943229433294342943529436294372943829439294402944129442294432944429445294462944729448294492945029451294522945329454294552945629457294582945929460294612946229463294642946529466294672946829469294702947129472294732947429475294762947729478294792948029481294822948329484294852948629487294882948929490294912949229493294942949529496294972949829499295002950129502295032950429505295062950729508295092951029511295122951329514295152951629517295182951929520295212952229523295242952529526295272952829529295302953129532295332953429535295362953729538295392954029541295422954329544295452954629547295482954929550295512955229553295542955529556295572955829559295602956129562295632956429565295662956729568295692957029571295722957329574295752957629577295782957929580295812958229583295842958529586295872958829589295902959129592295932959429595295962959729598295992960029601296022960329604296052960629607296082960929610296112961229613296142961529616296172961829619296202962129622296232962429625296262962729628296292963029631296322963329634296352963629637296382963929640296412964229643296442964529646296472964829649296502965129652296532965429655296562965729658296592966029661296622966329664296652966629667296682966929670296712967229673296742967529676296772967829679296802968129682296832968429685296862968729688296892969029691296922969329694296952969629697296982969929700297012970229703297042970529706297072970829709297102971129712297132971429715297162971729718297192972029721297222972329724297252972629727297282972929730297312973229733297342973529736297372973829739297402974129742297432974429745297462974729748297492975029751297522975329754297552975629757297582975929760297612976229763297642976529766297672976829769297702977129772297732977429775297762977729778297792978029781297822978329784297852978629787297882978929790297912979229793297942979529796297972979829799298002980129802298032980429805298062980729808298092981029811298122981329814298152981629817298182981929820298212982229823298242982529826298272982829829298302983129832298332983429835298362983729838298392984029841298422984329844298452984629847298482984929850298512985229853298542985529856298572985829859298602986129862298632986429865298662986729868298692987029871298722987329874298752987629877298782987929880298812988229883298842988529886298872988829889298902989129892298932989429895298962989729898298992990029901299022990329904299052990629907299082990929910299112991229913299142991529916299172991829919299202992129922299232992429925299262992729928299292993029931299322993329934299352993629937299382993929940299412994229943299442994529946299472994829949299502995129952299532995429955299562995729958299592996029961299622996329964299652996629967299682996929970299712997229973299742997529976299772997829979299802998129982299832998429985299862998729988299892999029991299922999329994299952999629997299982999930000300013000230003300043000530006300073000830009300103001130012300133001430015300163001730018300193002030021300223002330024300253002630027300283002930030300313003230033300343003530036300373003830039300403004130042300433004430045300463004730048300493005030051300523005330054300553005630057300583005930060300613006230063300643006530066300673006830069300703007130072300733007430075300763007730078300793008030081300823008330084300853008630087300883008930090300913009230093300943009530096300973009830099301003010130102301033010430105301063010730108301093011030111301123011330114301153011630117301183011930120301213012230123301243012530126301273012830129301303013130132301333013430135301363013730138301393014030141301423014330144301453014630147301483014930150301513015230153301543015530156301573015830159301603016130162301633016430165301663016730168301693017030171301723017330174301753017630177301783017930180301813018230183301843018530186301873018830189301903019130192301933019430195301963019730198301993020030201302023020330204302053020630207302083020930210302113021230213302143021530216302173021830219302203022130222302233022430225302263022730228302293023030231302323023330234302353023630237302383023930240302413024230243302443024530246302473024830249302503025130252302533025430255302563025730258302593026030261302623026330264302653026630267302683026930270302713027230273302743027530276302773027830279302803028130282302833028430285302863028730288302893029030291302923029330294302953029630297302983029930300303013030230303303043030530306303073030830309303103031130312303133031430315303163031730318303193032030321303223032330324303253032630327303283032930330303313033230333303343033530336303373033830339303403034130342303433034430345303463034730348303493035030351303523035330354303553035630357303583035930360303613036230363303643036530366303673036830369303703037130372303733037430375303763037730378303793038030381303823038330384303853038630387303883038930390303913039230393303943039530396303973039830399304003040130402304033040430405304063040730408304093041030411304123041330414304153041630417304183041930420304213042230423304243042530426304273042830429304303043130432304333043430435304363043730438304393044030441304423044330444304453044630447304483044930450304513045230453304543045530456304573045830459304603046130462304633046430465304663046730468304693047030471304723047330474304753047630477304783047930480304813048230483304843048530486304873048830489304903049130492304933049430495304963049730498304993050030501305023050330504305053050630507305083050930510305113051230513305143051530516305173051830519305203052130522305233052430525305263052730528305293053030531305323053330534305353053630537305383053930540305413054230543305443054530546305473054830549305503055130552305533055430555305563055730558305593056030561305623056330564305653056630567305683056930570305713057230573305743057530576305773057830579305803058130582305833058430585305863058730588305893059030591305923059330594305953059630597305983059930600306013060230603306043060530606306073060830609306103061130612306133061430615306163061730618306193062030621306223062330624306253062630627306283062930630306313063230633306343063530636306373063830639306403064130642306433064430645306463064730648306493065030651306523065330654306553065630657306583065930660306613066230663306643066530666306673066830669306703067130672306733067430675306763067730678306793068030681306823068330684306853068630687306883068930690306913069230693306943069530696306973069830699307003070130702307033070430705307063070730708307093071030711307123071330714307153071630717307183071930720307213072230723307243072530726307273072830729307303073130732307333073430735307363073730738307393074030741307423074330744307453074630747307483074930750307513075230753307543075530756307573075830759307603076130762307633076430765307663076730768307693077030771307723077330774307753077630777307783077930780307813078230783307843078530786307873078830789307903079130792307933079430795307963079730798307993080030801308023080330804308053080630807308083080930810308113081230813308143081530816308173081830819308203082130822308233082430825308263082730828308293083030831308323083330834308353083630837308383083930840308413084230843308443084530846308473084830849308503085130852308533085430855308563085730858308593086030861308623086330864308653086630867308683086930870308713087230873308743087530876308773087830879308803088130882308833088430885308863088730888308893089030891308923089330894308953089630897308983089930900309013090230903309043090530906309073090830909309103091130912309133091430915309163091730918309193092030921309223092330924309253092630927309283092930930309313093230933309343093530936309373093830939309403094130942309433094430945309463094730948309493095030951309523095330954309553095630957309583095930960309613096230963309643096530966309673096830969309703097130972309733097430975309763097730978309793098030981309823098330984309853098630987309883098930990309913099230993309943099530996309973099830999310003100131002310033100431005310063100731008310093101031011310123101331014310153101631017310183101931020310213102231023310243102531026310273102831029310303103131032310333103431035310363103731038310393104031041310423104331044310453104631047310483104931050310513105231053310543105531056310573105831059310603106131062310633106431065310663106731068310693107031071310723107331074310753107631077310783107931080310813108231083310843108531086310873108831089310903109131092310933109431095310963109731098310993110031101311023110331104311053110631107311083110931110311113111231113311143111531116311173111831119311203112131122311233112431125311263112731128311293113031131311323113331134311353113631137311383113931140311413114231143311443114531146311473114831149311503115131152311533115431155311563115731158311593116031161311623116331164311653116631167311683116931170311713117231173311743117531176311773117831179311803118131182311833118431185311863118731188311893119031191311923119331194311953119631197311983119931200312013120231203312043120531206312073120831209312103121131212312133121431215312163121731218312193122031221312223122331224312253122631227312283122931230312313123231233312343123531236312373123831239312403124131242312433124431245312463124731248312493125031251312523125331254312553125631257312583125931260312613126231263312643126531266312673126831269312703127131272312733127431275312763127731278312793128031281312823128331284312853128631287312883128931290312913129231293312943129531296312973129831299313003130131302313033130431305313063130731308313093131031311313123131331314313153131631317313183131931320313213132231323313243132531326313273132831329313303133131332313333133431335313363133731338313393134031341313423134331344313453134631347313483134931350313513135231353313543135531356313573135831359313603136131362313633136431365313663136731368313693137031371313723137331374313753137631377313783137931380313813138231383313843138531386313873138831389313903139131392313933139431395313963139731398313993140031401314023140331404314053140631407314083140931410314113141231413314143141531416314173141831419314203142131422314233142431425314263142731428314293143031431314323143331434314353143631437314383143931440314413144231443314443144531446314473144831449314503145131452314533145431455314563145731458314593146031461314623146331464314653146631467314683146931470314713147231473314743147531476314773147831479314803148131482314833148431485314863148731488314893149031491314923149331494314953149631497314983149931500315013150231503315043150531506315073150831509315103151131512315133151431515315163151731518315193152031521315223152331524315253152631527315283152931530315313153231533315343153531536315373153831539315403154131542315433154431545315463154731548315493155031551315523155331554315553155631557315583155931560315613156231563315643156531566315673156831569315703157131572315733157431575315763157731578315793158031581315823158331584315853158631587315883158931590315913159231593315943159531596315973159831599316003160131602316033160431605316063160731608316093161031611316123161331614316153161631617316183161931620316213162231623316243162531626316273162831629316303163131632316333163431635316363163731638316393164031641316423164331644316453164631647316483164931650316513165231653316543165531656316573165831659316603166131662316633166431665316663166731668316693167031671316723167331674316753167631677316783167931680316813168231683316843168531686316873168831689316903169131692316933169431695316963169731698316993170031701317023170331704317053170631707317083170931710317113171231713317143171531716317173171831719317203172131722317233172431725317263172731728317293173031731317323173331734317353173631737317383173931740317413174231743317443174531746317473174831749317503175131752317533175431755317563175731758317593176031761317623176331764317653176631767317683176931770317713177231773317743177531776317773177831779317803178131782317833178431785317863178731788317893179031791317923179331794317953179631797317983179931800318013180231803318043180531806318073180831809318103181131812318133181431815318163181731818318193182031821318223182331824318253182631827318283182931830318313183231833318343183531836318373183831839318403184131842318433184431845318463184731848318493185031851318523185331854318553185631857318583185931860318613186231863318643186531866318673186831869318703187131872318733187431875318763187731878318793188031881318823188331884318853188631887318883188931890318913189231893318943189531896318973189831899319003190131902319033190431905319063190731908319093191031911319123191331914319153191631917319183191931920319213192231923319243192531926319273192831929319303193131932319333193431935319363193731938319393194031941319423194331944319453194631947319483194931950319513195231953319543195531956319573195831959319603196131962319633196431965319663196731968319693197031971319723197331974319753197631977319783197931980319813198231983319843198531986319873198831989319903199131992319933199431995319963199731998319993200032001320023200332004320053200632007320083200932010320113201232013320143201532016320173201832019320203202132022320233202432025320263202732028320293203032031320323203332034320353203632037320383203932040320413204232043320443204532046320473204832049320503205132052320533205432055320563205732058320593206032061320623206332064320653206632067320683206932070320713207232073320743207532076320773207832079320803208132082320833208432085320863208732088320893209032091320923209332094320953209632097320983209932100321013210232103321043210532106321073210832109321103211132112321133211432115321163211732118321193212032121321223212332124321253212632127321283212932130321313213232133321343213532136321373213832139321403214132142321433214432145321463214732148321493215032151321523215332154321553215632157321583215932160321613216232163321643216532166321673216832169321703217132172321733217432175321763217732178321793218032181321823218332184321853218632187321883218932190321913219232193321943219532196321973219832199322003220132202322033220432205322063220732208322093221032211322123221332214322153221632217322183221932220322213222232223322243222532226322273222832229322303223132232322333223432235322363223732238322393224032241322423224332244322453224632247322483224932250322513225232253322543225532256322573225832259322603226132262322633226432265322663226732268322693227032271322723227332274322753227632277322783227932280322813228232283322843228532286322873228832289322903229132292322933229432295322963229732298322993230032301323023230332304323053230632307323083230932310323113231232313323143231532316323173231832319323203232132322323233232432325323263232732328323293233032331323323233332334323353233632337323383233932340323413234232343323443234532346323473234832349323503235132352323533235432355323563235732358323593236032361323623236332364323653236632367323683236932370323713237232373323743237532376323773237832379323803238132382323833238432385323863238732388323893239032391323923239332394323953239632397323983239932400324013240232403324043240532406324073240832409324103241132412324133241432415324163241732418324193242032421324223242332424324253242632427324283242932430324313243232433324343243532436324373243832439324403244132442324433244432445324463244732448324493245032451324523245332454324553245632457324583245932460324613246232463324643246532466324673246832469324703247132472324733247432475324763247732478324793248032481324823248332484324853248632487324883248932490324913249232493324943249532496324973249832499325003250132502325033250432505325063250732508325093251032511325123251332514325153251632517325183251932520325213252232523325243252532526325273252832529325303253132532325333253432535325363253732538325393254032541325423254332544325453254632547325483254932550325513255232553325543255532556325573255832559325603256132562325633256432565325663256732568325693257032571325723257332574325753257632577325783257932580325813258232583325843258532586325873258832589325903259132592325933259432595325963259732598325993260032601326023260332604326053260632607326083260932610326113261232613326143261532616326173261832619326203262132622326233262432625326263262732628326293263032631326323263332634326353263632637326383263932640326413264232643326443264532646326473264832649326503265132652326533265432655326563265732658326593266032661326623266332664326653266632667326683266932670326713267232673326743267532676326773267832679326803268132682326833268432685326863268732688326893269032691326923269332694326953269632697326983269932700327013270232703327043270532706327073270832709327103271132712327133271432715327163271732718327193272032721327223272332724327253272632727327283272932730327313273232733327343273532736327373273832739327403274132742327433274432745327463274732748327493275032751327523275332754327553275632757327583275932760327613276232763327643276532766327673276832769327703277132772327733277432775327763277732778327793278032781327823278332784327853278632787327883278932790327913279232793327943279532796327973279832799328003280132802328033280432805328063280732808328093281032811328123281332814328153281632817328183281932820328213282232823328243282532826328273282832829328303283132832328333283432835328363283732838328393284032841328423284332844328453284632847328483284932850328513285232853328543285532856328573285832859
  1. {
  2. This file is part of the Free Component Library (FCL)
  3. Copyright (c) 2018 by Michael Van Canneyt
  4. Unit tests for Pascal-to-Javascript converter class.
  5. See the file COPYING.FPC, included in this distribution,
  6. for details about the copyright.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  10. **********************************************************************
  11. Examples:
  12. ./testpas2js --suite=TTestModule.TestEmptyProgram
  13. ./testpas2js --suite=TTestModule.TestEmptyUnit
  14. }
  15. unit TCModules;
  16. {$mode objfpc}{$H+}
  17. interface
  18. uses
  19. Classes, SysUtils, fpcunit, testregistry, contnrs,
  20. jstree, jswriter, jsbase,
  21. PasTree, PScanner, PasResolver, PParser, PasResolveEval,
  22. FPPas2Js;
  23. const
  24. // default parser+scanner options
  25. po_tcmodules = po_Pas2js+[po_KeepScannerError];
  26. co_tcmodules = [];
  27. type
  28. TSrcMarkerKind = (
  29. mkLabel,
  30. mkResolverReference,
  31. mkDirectReference
  32. );
  33. PSrcMarker = ^TSrcMarker;
  34. TSrcMarker = record
  35. Kind: TSrcMarkerKind;
  36. Filename: string;
  37. Row: integer;
  38. StartCol, EndCol: integer; // token start, end column
  39. Identifier: string;
  40. Next: PSrcMarker;
  41. end;
  42. TSystemUnitPart = (
  43. supTObject,
  44. supTVarRec,
  45. supTypeInfo,
  46. supTInterfacedObject
  47. );
  48. TSystemUnitParts = set of TSystemUnitPart;
  49. { TTestHintMessage }
  50. TTestHintMessage = class
  51. public
  52. Id: int64;
  53. MsgType: TMessageType;
  54. MsgNumber: integer;
  55. Msg: string;
  56. SourcePos: TPasSourcePos;
  57. end;
  58. { TTestPasParser }
  59. TTestPasParser = Class(TPasParser)
  60. end;
  61. TOnFindUnit = function(const aUnitName: String): TPasModule of object;
  62. { TTestEnginePasResolver }
  63. TTestEnginePasResolver = class(TPas2JsResolver)
  64. private
  65. FFilename: string;
  66. FModule: TPasModule;
  67. FOnFindUnit: TOnFindUnit;
  68. FParser: TTestPasParser;
  69. FStreamResolver: TStreamResolver;
  70. FScanner: TPas2jsPasScanner;
  71. FSource: string;
  72. public
  73. destructor Destroy; override;
  74. function FindUnit(const AName, InFilename: String; NameExpr,
  75. InFileExpr: TPasExpr): TPasModule; override;
  76. procedure UsedInterfacesFinished(Section: TPasSection); override;
  77. property OnFindUnit: TOnFindUnit read FOnFindUnit write FOnFindUnit;
  78. property Filename: string read FFilename write FFilename;
  79. property StreamResolver: TStreamResolver read FStreamResolver write FStreamResolver;
  80. property Scanner: TPas2jsPasScanner read FScanner write FScanner;
  81. property Parser: TTestPasParser read FParser write FParser;
  82. property Source: string read FSource write FSource;
  83. property Module: TPasModule read FModule;
  84. end;
  85. { TCustomTestModule }
  86. TCustomTestModule = Class(TTestCase)
  87. private
  88. FConverter: TPasToJSConverter;
  89. FEngine: TTestEnginePasResolver;
  90. FExpectedErrorClass: ExceptClass;
  91. FExpectedErrorMsg: string;
  92. FExpectedErrorNumber: integer;
  93. FFilename: string;
  94. FFileResolver: TStreamResolver;
  95. FHub: TPas2JSResolverHub;
  96. FJSImplementationUses: TJSArrayLiteral;
  97. FJSInitBody: TJSFunctionBody;
  98. FJSImplentationUses: TJSArrayLiteral;
  99. FJSInterfaceUses: TJSArrayLiteral;
  100. FJSModule: TJSSourceElements;
  101. FJSModuleSrc: TJSSourceElements;
  102. FJSSource: TStringList;
  103. FModule: TPasModule;
  104. FJSModuleCallArgs: TJSArguments;
  105. FModules: TObjectList;// list of TTestEnginePasResolver
  106. FParser: TTestPasParser;
  107. FPasProgram: TPasProgram;
  108. FHintMsgs: TObjectList; // list of TTestHintMessage
  109. FHintMsgsGood: TFPList; // list of TTestHintMessage marked as expected
  110. FJSRegModuleCall: TJSCallExpression;
  111. FScanner: TPas2jsPasScanner;
  112. FSkipTests: boolean;
  113. FSource: TStringList;
  114. FFirstPasStatement: TPasImplBlock;
  115. FWithTypeInfo: boolean;
  116. {$IFDEF EnablePasTreeGlobalRefCount}
  117. FElementRefCountAtSetup: int64;
  118. {$ENDIF}
  119. function GetMsgCount: integer;
  120. function GetMsgs(Index: integer): TTestHintMessage;
  121. function GetResolverCount: integer;
  122. function GetResolvers(Index: integer): TTestEnginePasResolver;
  123. function OnPasResolverFindUnit(const aUnitName: String): TPasModule;
  124. procedure OnParserLog(Sender: TObject; const Msg: String);
  125. procedure OnPasResolverLog(Sender: TObject; const Msg: String);
  126. procedure OnScannerLog(Sender: TObject; const Msg: String);
  127. procedure SetWithTypeInfo(const AValue: boolean);
  128. protected
  129. procedure SetUp; override;
  130. function CreateConverter: TPasToJSConverter; virtual;
  131. function LoadUnit(const aUnitName: String): TPasModule;
  132. procedure InitScanner(aScanner: TPas2jsPasScanner); virtual;
  133. procedure TearDown; override;
  134. Procedure Add(Line: string); virtual;
  135. Procedure Add(const Lines: array of string);
  136. Procedure StartParsing; virtual;
  137. procedure ParseModuleQueue; virtual;
  138. procedure ParseModule; virtual;
  139. procedure ParseProgram; virtual;
  140. procedure ParseUnit; virtual;
  141. protected
  142. function FindModuleWithFilename(aFilename: string): TTestEnginePasResolver; virtual;
  143. function AddModule(aFilename: string): TTestEnginePasResolver; virtual;
  144. function AddModuleWithSrc(aFilename, Src: string): TTestEnginePasResolver; virtual;
  145. function AddModuleWithIntfImplSrc(aFilename, InterfaceSrc,
  146. ImplementationSrc: string): TTestEnginePasResolver; virtual;
  147. procedure AddSystemUnit(Parts: TSystemUnitParts = []); virtual;
  148. procedure StartProgram(NeedSystemUnit: boolean; SystemUnitParts: TSystemUnitParts = []); virtual;
  149. procedure StartUnit(NeedSystemUnit: boolean; SystemUnitParts: TSystemUnitParts = []); virtual;
  150. procedure ConvertModule; virtual;
  151. procedure ConvertProgram; virtual;
  152. procedure ConvertUnit; virtual;
  153. function ConvertJSModuleToString(El: TJSElement): string; virtual;
  154. procedure CheckDottedIdentifier(Msg: string; El: TJSElement; DottedName: string);
  155. function GetDottedIdentifier(El: TJSElement): string;
  156. procedure CheckSource(Msg,Statements: String; InitStatements: string = '';
  157. ImplStatements: string = ''); virtual;
  158. procedure CheckDiff(Msg, Expected, Actual: string); virtual;
  159. procedure CheckUnit(Filename, ExpectedSrc: string); virtual;
  160. procedure CheckHint(MsgType: TMessageType; MsgNumber: integer;
  161. Msg: string; Marker: PSrcMarker = nil); virtual;
  162. procedure CheckResolverUnexpectedHints(WithSourcePos: boolean = false); virtual;
  163. procedure SetExpectedScannerError(Msg: string; MsgNumber: integer);
  164. procedure SetExpectedParserError(Msg: string; MsgNumber: integer);
  165. procedure SetExpectedPasResolverError(Msg: string; MsgNumber: integer);
  166. procedure SetExpectedConverterError(Msg: string; MsgNumber: integer);
  167. function IsErrorExpected(E: Exception): boolean;
  168. procedure HandleScannerError(E: EScannerError);
  169. procedure HandleParserError(E: EParserError);
  170. procedure HandlePasResolveError(E: EPasResolve);
  171. procedure HandlePas2JSError(E: EPas2JS);
  172. procedure HandleException(E: Exception);
  173. procedure FailException(E: Exception);
  174. procedure WriteSources(const aFilename: string; aRow, aCol: integer);
  175. function IndexOfResolver(const Filename: string): integer;
  176. function GetResolver(const Filename: string): TTestEnginePasResolver;
  177. function GetDefaultNamespace: string;
  178. property PasProgram: TPasProgram Read FPasProgram;
  179. property Resolvers[Index: integer]: TTestEnginePasResolver read GetResolvers;
  180. property ResolverCount: integer read GetResolverCount;
  181. property Engine: TTestEnginePasResolver read FEngine;
  182. property Filename: string read FFilename;
  183. Property Module: TPasModule Read FModule;
  184. property FirstPasStatement: TPasImplBlock read FFirstPasStatement;
  185. property Converter: TPasToJSConverter read FConverter;
  186. property JSSource: TStringList read FJSSource;
  187. property JSModule: TJSSourceElements read FJSModule;
  188. property JSRegModuleCall: TJSCallExpression read FJSRegModuleCall;
  189. property JSModuleCallArgs: TJSArguments read FJSModuleCallArgs;
  190. property JSImplementationUses: TJSArrayLiteral read FJSImplementationUses;
  191. property JSInterfaceUses: TJSArrayLiteral read FJSInterfaceUses;
  192. property JSModuleSrc: TJSSourceElements read FJSModuleSrc;
  193. property JSInitBody: TJSFunctionBody read FJSInitBody;
  194. property ExpectedErrorClass: ExceptClass read FExpectedErrorClass write FExpectedErrorClass;
  195. property ExpectedErrorMsg: string read FExpectedErrorMsg write FExpectedErrorMsg;
  196. property ExpectedErrorNumber: integer read FExpectedErrorNumber write FExpectedErrorNumber;
  197. property SkipTests: boolean read FSkipTests write FSkipTests;
  198. public
  199. constructor Create; override;
  200. destructor Destroy; override;
  201. property Hub: TPas2JSResolverHub read FHub;
  202. property Source: TStringList read FSource;
  203. property FileResolver: TStreamResolver read FFileResolver;
  204. property Scanner: TPas2jsPasScanner read FScanner;
  205. property Parser: TTestPasParser read FParser;
  206. property MsgCount: integer read GetMsgCount;
  207. property Msgs[Index: integer]: TTestHintMessage read GetMsgs;
  208. property WithTypeInfo: boolean read FWithTypeInfo write SetWithTypeInfo;
  209. end;
  210. { TTestModule }
  211. TTestModule = class(TCustomTestModule)
  212. Published
  213. Procedure TestReservedWords;
  214. // program/units
  215. Procedure TestEmptyProgram;
  216. Procedure TestEmptyProgramUseStrict;
  217. Procedure TestEmptyUnit;
  218. Procedure TestEmptyUnitUseStrict;
  219. Procedure TestDottedUnitNames;
  220. Procedure TestDottedUnitNameImpl;
  221. Procedure TestDottedUnitExpr;
  222. Procedure Test_ModeFPCFail;
  223. Procedure Test_ModeSwitchCBlocksFail;
  224. Procedure TestUnit_UseSystem;
  225. Procedure TestUnit_Intf1Impl2Intf1;
  226. Procedure TestIncludeVersion;
  227. // vars/const
  228. Procedure TestVarInt;
  229. Procedure TestVarBaseTypes;
  230. Procedure TestBaseTypeSingleFail;
  231. Procedure TestBaseTypeExtendedFail;
  232. Procedure TestConstBaseTypes;
  233. Procedure TestUnitImplVars;
  234. Procedure TestUnitImplConsts;
  235. Procedure TestUnitImplRecord;
  236. Procedure TestRenameJSNameConflict;
  237. Procedure TestLocalConst;
  238. Procedure TestVarExternal;
  239. Procedure TestVarExternalOtherUnit;
  240. Procedure TestVarAbsoluteFail;
  241. Procedure TestConstExternal;
  242. // numbers
  243. Procedure TestDouble;
  244. Procedure TestInteger;
  245. Procedure TestIntegerRange;
  246. Procedure TestIntegerTypecasts;
  247. Procedure TestInteger_BitwiseShrNativeInt;
  248. Procedure TestInteger_BitwiseShlNativeInt;
  249. Procedure TestInteger_SystemFunc;
  250. Procedure TestCurrency;
  251. Procedure TestForBoolDo;
  252. Procedure TestForIntDo;
  253. Procedure TestForIntInDo;
  254. // strings
  255. Procedure TestCharConst;
  256. Procedure TestChar_Compare;
  257. Procedure TestChar_BuiltInProcs;
  258. Procedure TestStringConst;
  259. Procedure TestStringConstSurrogate;
  260. Procedure TestString_Length;
  261. Procedure TestString_Compare;
  262. Procedure TestString_SetLength;
  263. Procedure TestString_CharAt;
  264. Procedure TestStringHMinusFail;
  265. Procedure TestStr;
  266. Procedure TestBaseType_AnsiStringFail;
  267. Procedure TestBaseType_WideStringFail;
  268. Procedure TestBaseType_ShortStringFail;
  269. Procedure TestBaseType_RawByteStringFail;
  270. Procedure TestTypeShortstring_Fail;
  271. Procedure TestCharSet_Custom;
  272. Procedure TestWideChar_VarArg;
  273. Procedure TestForCharDo;
  274. Procedure TestForCharInDo;
  275. // alias types
  276. Procedure TestAliasTypeRef;
  277. Procedure TestTypeCast_BaseTypes;
  278. Procedure TestTypeCast_AliasBaseTypes;
  279. // functions
  280. Procedure TestEmptyProc;
  281. Procedure TestProcOneParam;
  282. Procedure TestFunctionWithoutParams;
  283. Procedure TestProcedureWithoutParams;
  284. Procedure TestPrgProcVar;
  285. Procedure TestProcTwoArgs;
  286. Procedure TestProc_DefaultValue;
  287. Procedure TestUnitProcVar;
  288. Procedure TestImplProc;
  289. Procedure TestFunctionResult;
  290. Procedure TestNestedProc;
  291. Procedure TestNestedProc_ResultString;
  292. Procedure TestForwardProc;
  293. Procedure TestNestedForwardProc;
  294. Procedure TestAssignFunctionResult;
  295. Procedure TestFunctionResultInCondition;
  296. Procedure TestFunctionResultInForLoop;
  297. Procedure TestFunctionResultInTypeCast;
  298. Procedure TestExit;
  299. Procedure TestExit_ResultInFinally;
  300. Procedure TestBreak;
  301. Procedure TestBreakAsVar;
  302. Procedure TestContinue;
  303. Procedure TestProc_External;
  304. Procedure TestProc_ExternalOtherUnit;
  305. Procedure TestProc_Asm;
  306. Procedure TestProc_Assembler;
  307. Procedure TestProc_VarParam;
  308. Procedure TestProc_VarParamString;
  309. Procedure TestProc_VarParamV;
  310. Procedure TestProc_Overload;
  311. Procedure TestProc_OverloadForward;
  312. Procedure TestProc_OverloadIntfImpl;
  313. Procedure TestProc_OverloadNested;
  314. Procedure TestProc_OverloadNestedForward;
  315. Procedure TestProc_OverloadUnitCycle;
  316. Procedure TestProc_Varargs;
  317. Procedure TestProc_ConstOrder;
  318. Procedure TestProc_DuplicateConst;
  319. Procedure TestProc_LocalVarAbsolute;
  320. Procedure TestProc_LocalVarInit;
  321. Procedure TestProc_ReservedWords;
  322. Procedure TestProc_ConstRefWord;
  323. // anonymous functions
  324. Procedure TestAnonymousProc_Assign_ObjFPC;
  325. Procedure TestAnonymousProc_Assign_Delphi;
  326. Procedure TestAnonymousProc_Arg;
  327. Procedure TestAnonymousProc_Typecast;
  328. Procedure TestAnonymousProc_With;
  329. Procedure TestAnonymousProc_ExceptOn;
  330. Procedure TestAnonymousProc_Nested;
  331. Procedure TestAnonymousProc_NestedAssignResult;
  332. Procedure TestAnonymousProc_Class;
  333. Procedure TestAnonymousProc_ForLoop;
  334. Procedure TestAnonymousProc_AsmDelphi;
  335. // enums, sets
  336. Procedure TestEnum_Name;
  337. Procedure TestEnum_Number;
  338. Procedure TestEnum_ConstFail;
  339. Procedure TestEnum_Functions;
  340. Procedure TestEnumRg_Functions;
  341. Procedure TestEnum_AsParams;
  342. Procedure TestEnumRange_Array;
  343. Procedure TestEnum_ForIn;
  344. Procedure TestEnum_ScopedNumber;
  345. Procedure TestEnum_InFunction;
  346. Procedure TestSet_Enum;
  347. Procedure TestSet_Operators;
  348. Procedure TestSet_Operator_In;
  349. Procedure TestSet_Functions;
  350. Procedure TestSet_PassAsArgClone;
  351. Procedure TestSet_AsParams;
  352. Procedure TestSet_Property;
  353. Procedure TestSet_EnumConst;
  354. Procedure TestSet_IntConst;
  355. Procedure TestSet_AnonymousEnumType;
  356. Procedure TestSet_AnonymousEnumTypeChar; // ToDo
  357. Procedure TestSet_ConstEnum;
  358. Procedure TestSet_ConstChar;
  359. Procedure TestSet_ConstInt;
  360. Procedure TestSet_InFunction;
  361. Procedure TestSet_ForIn;
  362. // statements
  363. Procedure TestNestBegin;
  364. Procedure TestIncDec;
  365. Procedure TestLoHiFpcMode;
  366. Procedure TestLoHiDelphiMode;
  367. Procedure TestAssignments;
  368. Procedure TestArithmeticOperators1;
  369. Procedure TestLogicalOperators;
  370. Procedure TestBitwiseOperators;
  371. Procedure TestBitwiseOperatorsLongword;
  372. Procedure TestFunctionInt;
  373. Procedure TestFunctionString;
  374. Procedure TestIfThen;
  375. Procedure TestForLoop;
  376. Procedure TestForLoopInsideFunction;
  377. Procedure TestForLoop_ReadVarAfter;
  378. Procedure TestForLoop_Nested;
  379. Procedure TestRepeatUntil;
  380. Procedure TestAsmBlock;
  381. Procedure TestAsmPas_Impl; // ToDo
  382. Procedure TestTryFinally;
  383. Procedure TestTryExcept;
  384. Procedure TestTryExcept_ReservedWords;
  385. Procedure TestIfThenRaiseElse;
  386. Procedure TestCaseOf;
  387. Procedure TestCaseOf_UseSwitch;
  388. Procedure TestCaseOfNoElse;
  389. Procedure TestCaseOfNoElse_UseSwitch;
  390. Procedure TestCaseOfRange;
  391. Procedure TestCaseOfString;
  392. Procedure TestCaseOfChar;
  393. Procedure TestCaseOfExternalClassConst;
  394. Procedure TestDebugger;
  395. // arrays
  396. Procedure TestArray_Dynamic;
  397. Procedure TestArray_Dynamic_Nil;
  398. Procedure TestArray_DynMultiDimensional;
  399. Procedure TestArray_DynamicAssign;
  400. Procedure TestArray_StaticInt;
  401. Procedure TestArray_StaticBool;
  402. Procedure TestArray_StaticChar;
  403. Procedure TestArray_StaticMultiDim;
  404. Procedure TestArray_StaticInFunction;
  405. Procedure TestArray_StaticMultiDimEqualNotImplemented;
  406. Procedure TestArrayOfRecord;
  407. Procedure TestArray_StaticRecord;
  408. Procedure TestArrayOfSet;
  409. Procedure TestArray_DynAsParam;
  410. Procedure TestArray_StaticAsParam;
  411. Procedure TestArrayElement_AsParams;
  412. Procedure TestArrayElementFromFuncResult_AsParams;
  413. Procedure TestArrayEnumTypeRange;
  414. Procedure TestArray_SetLengthOutArg;
  415. Procedure TestArray_SetLengthProperty;
  416. Procedure TestArray_SetLengthMultiDim;
  417. Procedure TestArray_SetLengthDynOfStatic;
  418. Procedure TestArray_OpenArrayOfString;
  419. Procedure TestArray_ArrayOfCharAssignString; // ToDo
  420. Procedure TestArray_ConstRef;
  421. Procedure TestArray_Concat;
  422. Procedure TestArray_Copy;
  423. Procedure TestArray_InsertDelete;
  424. Procedure TestArray_DynArrayConstObjFPC;
  425. Procedure TestArray_DynArrayConstDelphi;
  426. Procedure TestArray_ArrayLitAsParam;
  427. Procedure TestArray_ArrayLitMultiDimAsParam;
  428. Procedure TestArray_ArrayLitStaticAsParam;
  429. Procedure TestArray_ForInArrOfString;
  430. Procedure TestExternalClass_TypeCastArrayToExternalClass;
  431. Procedure TestExternalClass_TypeCastArrayFromExternalClass;
  432. Procedure TestArrayOfConst_TVarRec;
  433. Procedure TestArrayOfConst_PassBaseTypes;
  434. Procedure TestArrayOfConst_PassObj;
  435. // record
  436. Procedure TestRecord_Empty;
  437. Procedure TestRecord_Var;
  438. Procedure TestRecord_VarExternal;
  439. Procedure TestRecord_WithDo;
  440. Procedure TestRecord_Assign;
  441. Procedure TestRecord_AsParams;
  442. Procedure TestRecord_ConstRef;
  443. Procedure TestRecordElement_AsParams;
  444. Procedure TestRecordElementFromFuncResult_AsParams;
  445. Procedure TestRecordElementFromWith_AsParams;
  446. Procedure TestRecord_Equal;
  447. Procedure TestRecord_JSValue;
  448. Procedure TestRecord_VariantFail;
  449. Procedure TestRecord_FieldArray;
  450. Procedure TestRecord_Const;
  451. Procedure TestRecord_TypecastFail;
  452. Procedure TestRecord_InFunction;
  453. Procedure TestRecord_AnonymousFail;
  454. // advanced record
  455. Procedure TestAdvRecord_Function;
  456. Procedure TestAdvRecord_Property;
  457. Procedure TestAdvRecord_PropertyDefault;
  458. Procedure TestAdvRecord_Property_ClassMethod;
  459. Procedure TestAdvRecord_Const;
  460. Procedure TestAdvRecord_ExternalField;
  461. Procedure TestAdvRecord_SubRecord;
  462. Procedure TestAdvRecord_SubClass;
  463. Procedure TestAdvRecord_SubInterfaceFail;
  464. Procedure TestAdvRecord_Constructor;
  465. Procedure TestAdvRecord_ClassConstructor_Program;
  466. Procedure TestAdvRecord_ClassConstructor_Unit;
  467. // classes
  468. Procedure TestClass_TObjectDefaultConstructor;
  469. Procedure TestClass_TObjectConstructorWithParams;
  470. Procedure TestClass_TObjectConstructorWithDefaultParam;
  471. Procedure TestClass_Var;
  472. Procedure TestClass_Method;
  473. Procedure TestClass_Implementation;
  474. Procedure TestClass_Inheritance;
  475. Procedure TestClass_TypeAlias;
  476. Procedure TestClass_AbstractMethod;
  477. Procedure TestClass_CallInherited_ProcNoParams;
  478. Procedure TestClass_CallInherited_WithParams;
  479. Procedure TestClasS_CallInheritedConstructor;
  480. Procedure TestClass_ClassVar_Assign;
  481. Procedure TestClass_CallClassMethod;
  482. Procedure TestClass_Property;
  483. Procedure TestClass_Property_ClassMethod;
  484. Procedure TestClass_Property_Indexed;
  485. Procedure TestClass_Property_IndexSpec;
  486. Procedure TestClass_PropertyOfTypeArray;
  487. Procedure TestClass_PropertyDefault;
  488. Procedure TestClass_PropertyDefault_TypecastToOtherDefault;
  489. //Procedure TestClass_PropertyDefault;
  490. Procedure TestClass_PropertyOverride;
  491. Procedure TestClass_PropertyIncVisibility;
  492. Procedure TestClass_Assigned;
  493. Procedure TestClass_WithClassDoCreate;
  494. Procedure TestClass_WithClassInstDoProperty;
  495. Procedure TestClass_WithClassInstDoPropertyWithParams;
  496. Procedure TestClass_WithClassInstDoFunc;
  497. Procedure TestClass_TypeCast;
  498. Procedure TestClass_TypeCastUntypedParam;
  499. Procedure TestClass_Overloads;
  500. Procedure TestClass_OverloadsAncestor;
  501. Procedure TestClass_OverloadConstructor;
  502. Procedure TestClass_OverloadDelphiOverride;
  503. Procedure TestClass_ReintroduceVarDelphi;
  504. Procedure TestClass_ReintroducedVar;
  505. Procedure TestClass_RaiseDescendant;
  506. Procedure TestClass_ExternalMethod;
  507. Procedure TestClass_ExternalVirtualNameMismatchFail;
  508. Procedure TestClass_ExternalOverrideFail;
  509. Procedure TestClass_ExternalVar;
  510. Procedure TestClass_Const;
  511. Procedure TestClass_ConstEnum;
  512. Procedure TestClass_LocalConstDuplicate_Prg;
  513. Procedure TestClass_LocalConstDuplicate_Unit;
  514. // ToDo: Procedure TestAdvRecord_LocalConstDuplicate;
  515. Procedure TestClass_LocalVarSelfFail;
  516. Procedure TestClass_ArgSelfFail;
  517. Procedure TestClass_NestedProcSelf;
  518. Procedure TestClass_NestedProcSelf2;
  519. Procedure TestClass_NestedProcClassSelf;
  520. Procedure TestClass_NestedProcCallInherited;
  521. Procedure TestClass_TObjectFree;
  522. Procedure TestClass_TObjectFree_VarArg;
  523. Procedure TestClass_TObjectFreeNewInstance;
  524. Procedure TestClass_TObjectFreeLowerCase;
  525. Procedure TestClass_TObjectFreeFunctionFail;
  526. Procedure TestClass_TObjectFreePropertyFail;
  527. Procedure TestClass_ForIn;
  528. Procedure TestClass_DispatchMessage;
  529. Procedure TestClass_Message_DuplicateIntFail;
  530. Procedure TestClass_DispatchMessage_WrongFieldNameFail;
  531. // class of
  532. Procedure TestClassOf_Create;
  533. Procedure TestClassOf_Call;
  534. Procedure TestClassOf_Assign;
  535. Procedure TestClassOf_Is;
  536. Procedure TestClassOf_Compare;
  537. Procedure TestClassOf_ClassVar;
  538. Procedure TestClassOf_ClassMethod;
  539. Procedure TestClassOf_ClassProperty;
  540. Procedure TestClassOf_ClassMethodSelf;
  541. Procedure TestClassOf_TypeCast;
  542. Procedure TestClassOf_ImplicitFunctionCall;
  543. Procedure TestClassOf_Const;
  544. // nested class
  545. Procedure TestNestedClass_Alias;
  546. Procedure TestNestedClass_Record;
  547. Procedure TestNestedClass_Class;
  548. // external class
  549. Procedure TestExternalClass_Var;
  550. Procedure TestExternalClass_Const;
  551. Procedure TestExternalClass_Dollar;
  552. Procedure TestExternalClass_DuplicateVarFail;
  553. Procedure TestExternalClass_Method;
  554. Procedure TestExternalClass_ClassMethod;
  555. Procedure TestExternalClass_ClassMethodStatic;
  556. Procedure TestExternalClass_FunctionResultInTypeCast;
  557. Procedure TestExternalClass_NonExternalOverride;
  558. Procedure TestExternalClass_OverloadHint;
  559. Procedure TestExternalClass_SameNamePublishedProperty;
  560. Procedure TestExternalClass_Property;
  561. Procedure TestExternalClass_PropertyDate;
  562. Procedure TestExternalClass_ClassProperty;
  563. Procedure TestExternalClass_ClassOf;
  564. Procedure TestExternalClass_ClassOtherUnit;
  565. Procedure TestExternalClass_Is;
  566. Procedure TestExternalClass_As;
  567. Procedure TestExternalClass_DestructorFail;
  568. Procedure TestExternalClass_New;
  569. Procedure TestExternalClass_ClassOf_New;
  570. Procedure TestExternalClass_FuncClassOf_New;
  571. Procedure TestExternalClass_New_PasClassFail;
  572. Procedure TestExternalClass_New_PasClassBracketsFail;
  573. Procedure TestExternalClass_NewExtName;
  574. Procedure TestExternalClass_Constructor;
  575. Procedure TestExternalClass_ConstructorBrackets;
  576. Procedure TestExternalClass_LocalConstSameName;
  577. Procedure TestExternalClass_ReintroduceOverload;
  578. Procedure TestExternalClass_Inherited;
  579. Procedure TestExternalClass_PascalAncestorFail;
  580. Procedure TestExternalClass_NewInstance;
  581. Procedure TestExternalClass_NewInstance_NonVirtualFail;
  582. Procedure TestExternalClass_NewInstance_FirstParamNotString_Fail;
  583. Procedure TestExternalClass_NewInstance_SecondParamTyped_Fail;
  584. Procedure TestExternalClass_JSFunctionPasDescendant;
  585. Procedure TestExternalClass_PascalProperty;
  586. Procedure TestExternalClass_TypeCastToRootClass;
  587. Procedure TestExternalClass_TypeCastToJSObject;
  588. Procedure TestExternalClass_TypeCastStringToExternalString;
  589. Procedure TestExternalClass_TypeCastToJSFunction;
  590. Procedure TestExternalClass_TypeCastDelphiUnrelated;
  591. Procedure TestExternalClass_CallClassFunctionOfInstanceFail;
  592. Procedure TestExternalClass_BracketAccessor;
  593. Procedure TestExternalClass_BracketAccessor_Call;
  594. Procedure TestExternalClass_BracketAccessor_2ParamsFail;
  595. Procedure TestExternalClass_BracketAccessor_ReadOnly;
  596. Procedure TestExternalClass_BracketAccessor_WriteOnly;
  597. Procedure TestExternalClass_BracketAccessor_MultiType;
  598. Procedure TestExternalClass_BracketAccessor_Index;
  599. Procedure TestExternalClass_ForInJSObject;
  600. Procedure TestExternalClass_ForInJSArray;
  601. Procedure TestExternalClass_IncompatibleArgDuplicateIdentifier;
  602. // class interfaces
  603. Procedure TestClassInterface_Corba;
  604. Procedure TestClassInterface_ProcExternalFail;
  605. Procedure TestClassInterface_Overloads;
  606. Procedure TestClassInterface_DuplicateGUIInIntfListFail;
  607. Procedure TestClassInterface_DuplicateGUIInAncestorFail;
  608. Procedure TestClassInterface_AncestorImpl;
  609. Procedure TestClassInterface_ImplReintroduce;
  610. Procedure TestClassInterface_MethodResolution;
  611. Procedure TestClassInterface_AncestorMoreInterfaces;
  612. Procedure TestClassInterface_MethodOverride;
  613. Procedure TestClassInterface_Corba_Delegation;
  614. Procedure TestClassInterface_Corba_DelegationStatic;
  615. Procedure TestClassInterface_Corba_Operators;
  616. Procedure TestClassInterface_Corba_Args;
  617. Procedure TestClassInterface_Corba_ForIn;
  618. Procedure TestClassInterface_COM_AssignVar;
  619. Procedure TestClassInterface_COM_AssignArg;
  620. Procedure TestClassInterface_COM_FunctionResult;
  621. Procedure TestClassInterface_COM_InheritedFuncResult;
  622. Procedure TestClassInterface_COM_IsAsTypeCasts;
  623. Procedure TestClassInterface_COM_PassAsArg;
  624. Procedure TestClassInterface_COM_PassToUntypedParam;
  625. Procedure TestClassInterface_COM_FunctionInExpr;
  626. Procedure TestClassInterface_COM_Property;
  627. Procedure TestClassInterface_COM_IntfProperty;
  628. Procedure TestClassInterface_COM_Delegation;
  629. Procedure TestClassInterface_COM_With;
  630. Procedure TestClassInterface_COM_ForIn;
  631. Procedure TestClassInterface_COM_ArrayOfIntfFail;
  632. Procedure TestClassInterface_COM_RecordIntfFail;
  633. Procedure TestClassInterface_COM_UnitInitialization;
  634. Procedure TestClassInterface_GUID;
  635. Procedure TestClassInterface_GUIDProperty;
  636. // helpers
  637. Procedure TestClassHelper_ClassVar;
  638. Procedure TestClassHelper_Method_AccessInstanceFields;
  639. Procedure TestClassHelper_Method_Call;
  640. Procedure TestClassHelper_Method_Nested_Call;
  641. Procedure TestClassHelper_ClassMethod_Call;
  642. Procedure TestClassHelper_ClassOf;
  643. Procedure TestClassHelper_MethodRefObjFPC;
  644. Procedure TestClassHelper_Constructor;
  645. Procedure TestClassHelper_InheritedObjFPC;
  646. Procedure TestClassHelper_Property;
  647. Procedure TestClassHelper_Property_Array;
  648. Procedure TestClassHelper_Property_Array_Default;
  649. Procedure TestClassHelper_Property_Array_DefaultDefault;
  650. Procedure TestClassHelper_ClassProperty;
  651. Procedure TestClassHelper_ClassPropertyStatic;
  652. Procedure TestClassHelper_ClassProperty_Array;
  653. Procedure TestClassHelper_ForIn;
  654. Procedure TestClassHelper_PassProperty;
  655. Procedure TestExtClassHelper_ClassVar;
  656. Procedure TestExtClassHelper_Method_Call;
  657. Procedure TestExtClassHelper_ClassMethod_MissingStatic;
  658. Procedure TestRecordHelper_ClassVar;
  659. Procedure TestRecordHelper_Method_Call;
  660. Procedure TestRecordHelper_Constructor;
  661. Procedure TestTypeHelper_ClassVar;
  662. Procedure TestTypeHelper_PassResultElement;
  663. Procedure TestTypeHelper_PassArgs;
  664. Procedure TestTypeHelper_PassVarConst;
  665. Procedure TestTypeHelper_PassFuncResult;
  666. Procedure TestTypeHelper_PassPropertyField;
  667. Procedure TestTypeHelper_PassPropertyGetter;
  668. Procedure TestTypeHelper_PassClassPropertyField;
  669. Procedure TestTypeHelper_PassClassPropertyGetterStatic;
  670. Procedure TestTypeHelper_PassClassPropertyGetterNonStatic;
  671. Procedure TestTypeHelper_Property;
  672. Procedure TestTypeHelper_Property_Array;
  673. Procedure TestTypeHelper_ClassProperty;
  674. Procedure TestTypeHelper_ClassProperty_Array;
  675. Procedure TestTypeHelper_ClassMethod;
  676. Procedure TestTypeHelper_ExtClassMethodFail;
  677. Procedure TestTypeHelper_Constructor;
  678. Procedure TestTypeHelper_Word;
  679. Procedure TestTypeHelper_Boolean;
  680. Procedure TestTypeHelper_WordBool;
  681. Procedure TestTypeHelper_Double;
  682. Procedure TestTypeHelper_NativeInt;
  683. Procedure TestTypeHelper_StringChar;
  684. Procedure TestTypeHelper_JSValue;
  685. Procedure TestTypeHelper_Array;
  686. Procedure TestTypeHelper_EnumType;
  687. Procedure TestTypeHelper_SetType;
  688. Procedure TestTypeHelper_InterfaceType;
  689. Procedure TestTypeHelper_NestedSelf;
  690. // proc types
  691. Procedure TestProcType;
  692. Procedure TestProcType_Arg;
  693. Procedure TestProcType_FunctionFPC;
  694. Procedure TestProcType_FunctionDelphi;
  695. Procedure TestProcType_ProcedureDelphi;
  696. Procedure TestProcType_AsParam;
  697. Procedure TestProcType_MethodFPC;
  698. Procedure TestProcType_MethodDelphi;
  699. Procedure TestProcType_PropertyFPC;
  700. Procedure TestProcType_PropertyDelphi;
  701. Procedure TestProcType_WithClassInstDoPropertyFPC;
  702. Procedure TestProcType_Nested;
  703. Procedure TestProcType_NestedOfObject;
  704. Procedure TestProcType_ReferenceToProc;
  705. Procedure TestProcType_ReferenceToMethod;
  706. Procedure TestProcType_Typecast;
  707. Procedure TestProcType_PassProcToUntyped;
  708. Procedure TestProcType_PassProcToArray;
  709. Procedure TestProcType_SafeCallObjFPC;
  710. Procedure TestProcType_SafeCallDelphi;
  711. // pointer
  712. Procedure TestPointer;
  713. Procedure TestPointer_Proc;
  714. Procedure TestPointer_AssignRecordFail;
  715. Procedure TestPointer_AssignStaticArrayFail;
  716. Procedure TestPointer_TypeCastJSValueToPointer;
  717. Procedure TestPointer_NonRecordFail;
  718. Procedure TestPointer_AnonymousArgTypeFail;
  719. Procedure TestPointer_AnonymousVarTypeFail;
  720. Procedure TestPointer_AnonymousResultTypeFail;
  721. Procedure TestPointer_AddrOperatorFail;
  722. Procedure TestPointer_ArrayParamsFail;
  723. Procedure TestPointer_PointerAddFail;
  724. Procedure TestPointer_IncPointerFail;
  725. Procedure TestPointer_Record;
  726. Procedure TestPointer_RecordArg;
  727. // jsvalue
  728. Procedure TestJSValue_AssignToJSValue;
  729. Procedure TestJSValue_TypeCastToBaseType;
  730. Procedure TestJSValue_TypecastToJSValue;
  731. Procedure TestJSValue_Equal;
  732. Procedure TestJSValue_If;
  733. Procedure TestJSValue_Not;
  734. Procedure TestJSValue_Enum;
  735. Procedure TestJSValue_ClassInstance;
  736. Procedure TestJSValue_ClassOf;
  737. Procedure TestJSValue_ArrayOfJSValue;
  738. Procedure TestJSValue_ArrayLit;
  739. Procedure TestJSValue_Params;
  740. Procedure TestJSValue_UntypedParam;
  741. Procedure TestJSValue_FuncResultType;
  742. Procedure TestJSValue_ProcType_Assign;
  743. Procedure TestJSValue_ProcType_Equal;
  744. Procedure TestJSValue_ProcType_Param;
  745. Procedure TestJSValue_AssignToPointerFail;
  746. Procedure TestJSValue_OverloadDouble;
  747. Procedure TestJSValue_OverloadNativeInt;
  748. Procedure TestJSValue_OverloadWord;
  749. Procedure TestJSValue_OverloadString;
  750. Procedure TestJSValue_OverloadChar;
  751. Procedure TestJSValue_OverloadPointer;
  752. Procedure TestJSValue_ForIn;
  753. // RTTI
  754. Procedure TestRTTI_IntRange;
  755. Procedure TestRTTI_Double;
  756. Procedure TestRTTI_ProcType;
  757. Procedure TestRTTI_ProcType_ArgFromOtherUnit;
  758. Procedure TestRTTI_EnumAndSetType;
  759. Procedure TestRTTI_EnumRange;
  760. Procedure TestRTTI_AnonymousEnumType;
  761. Procedure TestRTTI_StaticArray;
  762. Procedure TestRTTI_DynArray;
  763. Procedure TestRTTI_ArrayNestedAnonymous;
  764. Procedure TestRTTI_PublishedMethodOverloadFail;
  765. Procedure TestRTTI_PublishedMethodExternalFail;
  766. Procedure TestRTTI_PublishedClassPropertyFail;
  767. Procedure TestRTTI_PublishedClassFieldFail;
  768. Procedure TestRTTI_PublishedFieldExternalFail;
  769. Procedure TestRTTI_Class_Field;
  770. Procedure TestRTTI_Class_Method;
  771. Procedure TestRTTI_Class_MethodArgFlags;
  772. Procedure TestRTTI_Class_Property;
  773. Procedure TestRTTI_Class_PropertyParams;
  774. Procedure TestRTTI_Class_OtherUnit_TypeAlias;
  775. Procedure TestRTTI_Class_OmitRTTI;
  776. Procedure TestRTTI_IndexModifier;
  777. Procedure TestRTTI_StoredModifier;
  778. Procedure TestRTTI_DefaultValue;
  779. Procedure TestRTTI_DefaultValueSet;
  780. Procedure TestRTTI_DefaultValueRangeType;
  781. Procedure TestRTTI_DefaultValueInherit;
  782. Procedure TestRTTI_OverrideMethod;
  783. Procedure TestRTTI_ReintroduceMethod;
  784. Procedure TestRTTI_OverloadProperty;
  785. // ToDo: array argument
  786. Procedure TestRTTI_ClassForward;
  787. Procedure TestRTTI_ClassOf;
  788. Procedure TestRTTI_Record;
  789. Procedure TestRTTI_RecordAnonymousArray;
  790. Procedure TestRTTI_LocalTypes;
  791. Procedure TestRTTI_TypeInfo_BaseTypes;
  792. Procedure TestRTTI_TypeInfo_Type_BaseTypes;
  793. Procedure TestRTTI_TypeInfo_LocalFail;
  794. Procedure TestRTTI_TypeInfo_ExtTypeInfoClasses1;
  795. Procedure TestRTTI_TypeInfo_ExtTypeInfoClasses2;
  796. Procedure TestRTTI_TypeInfo_ExtTypeInfoClasses3;
  797. Procedure TestRTTI_TypeInfo_FunctionClassType;
  798. Procedure TestRTTI_TypeInfo_MixedUnits_PointerAndClass;
  799. Procedure TestRTTI_Interface_Corba;
  800. Procedure TestRTTI_Interface_COM;
  801. Procedure TestRTTI_ClassHelper;
  802. Procedure TestRTTI_ExternalClass;
  803. Procedure TestRTTI_Unit;
  804. // Resourcestring
  805. Procedure TestResourcestringProgram;
  806. Procedure TestResourcestringUnit;
  807. Procedure TestResourcestringImplementation;
  808. // Attributes
  809. Procedure TestAttributes_Members;
  810. Procedure TestAttributes_Types;
  811. Procedure TestAttributes_HelperConstructor_Fail;
  812. // Assertions, checks
  813. procedure TestAssert;
  814. procedure TestAssert_SysUtils;
  815. procedure TestObjectChecks;
  816. procedure TestOverflowChecks_Int;
  817. procedure TestRangeChecks_AssignInt;
  818. procedure TestRangeChecks_AssignIntRange;
  819. procedure TestRangeChecks_AssignEnum;
  820. procedure TestRangeChecks_AssignEnumRange;
  821. procedure TestRangeChecks_AssignChar;
  822. procedure TestRangeChecks_AssignCharRange;
  823. procedure TestRangeChecks_ArrayIndex;
  824. procedure TestRangeChecks_ArrayOfRecIndex;
  825. procedure TestRangeChecks_StringIndex;
  826. procedure TestRangeChecks_TypecastInt;
  827. procedure TestRangeChecks_TypeHelperInt;
  828. // Async/AWait
  829. Procedure TestAsync_Proc;
  830. Procedure TestAsync_CallResultIsPromise;
  831. Procedure TestAsync_ConstructorFail;
  832. Procedure TestAsync_PropertyGetterFail;
  833. Procedure TestAwait_NonPromiseWithTypeFail;
  834. Procedure TestAwait_AsyncCallTypeMismatch;
  835. Procedure TestAWait_OutsideAsyncFail;
  836. Procedure TestAWait_Result;
  837. Procedure TestAWait_ExternalClassPromise;
  838. Procedure TestAsync_AnonymousProc;
  839. Procedure TestAsync_ProcType;
  840. Procedure TestAsync_ProcTypeAsyncModMismatchFail;
  841. Procedure TestAsync_Inherited;
  842. Procedure TestAsync_ClassInterface;
  843. end;
  844. function LinesToStr(Args: array of const): string;
  845. function ExtractFileUnitName(aFilename: string): string;
  846. function JSToStr(El: TJSElement): string;
  847. function CheckSrcDiff(Expected, Actual: string; out Msg: string): boolean;
  848. implementation
  849. function LinesToStr(Args: array of const): string;
  850. var
  851. s: String;
  852. i: Integer;
  853. begin
  854. s:='';
  855. for i:=Low(Args) to High(Args) do
  856. case Args[i].VType of
  857. vtChar: s += Args[i].VChar+LineEnding;
  858. vtString: s += Args[i].VString^+LineEnding;
  859. vtPChar: s += Args[i].VPChar+LineEnding;
  860. vtWideChar: s += AnsiString(Args[i].VWideChar)+LineEnding;
  861. vtPWideChar: s += AnsiString(Args[i].VPWideChar)+LineEnding;
  862. vtAnsiString: s += AnsiString(Args[i].VAnsiString)+LineEnding;
  863. vtWidestring: s += AnsiString(WideString(Args[i].VWideString))+LineEnding;
  864. vtUnicodeString:s += AnsiString(UnicodeString(Args[i].VUnicodeString))+LineEnding;
  865. end;
  866. Result:=s;
  867. end;
  868. function ExtractFileUnitName(aFilename: string): string;
  869. var
  870. p: Integer;
  871. begin
  872. Result:=ExtractFileName(aFilename);
  873. if Result='' then exit;
  874. for p:=length(Result) downto 1 do
  875. case Result[p] of
  876. '/','\': exit;
  877. '.':
  878. begin
  879. Delete(Result,p,length(Result));
  880. exit;
  881. end;
  882. end;
  883. end;
  884. function JSToStr(El: TJSElement): string;
  885. var
  886. aWriter: TBufferWriter;
  887. aJSWriter: TJSWriter;
  888. begin
  889. aJSWriter:=nil;
  890. aWriter:=TBufferWriter.Create(1000);
  891. try
  892. aJSWriter:=TJSWriter.Create(aWriter);
  893. aJSWriter.IndentSize:=2;
  894. aJSWriter.WriteJS(El);
  895. Result:=aWriter.AsString;
  896. finally
  897. aJSWriter.Free;
  898. aWriter.Free;
  899. end;
  900. end;
  901. function CheckSrcDiff(Expected, Actual: string; out Msg: string): boolean;
  902. // search diff, ignore changes in spaces
  903. const
  904. SpaceChars = [#9,#10,#13,' '];
  905. var
  906. ExpectedP, ActualP: PChar;
  907. function FindLineEnd(p: PChar): PChar;
  908. begin
  909. Result:=p;
  910. while not (Result^ in [#0,#10,#13]) do inc(Result);
  911. end;
  912. function FindLineStart(p, MinP: PChar): PChar;
  913. begin
  914. while (p>MinP) and not (p[-1] in [#10,#13]) do dec(p);
  915. Result:=p;
  916. end;
  917. procedure SkipLineEnd(var p: PChar);
  918. begin
  919. if p^ in [#10,#13] then
  920. begin
  921. if (p[1] in [#10,#13]) and (p^<>p[1]) then
  922. inc(p,2)
  923. else
  924. inc(p);
  925. end;
  926. end;
  927. procedure DiffFound;
  928. var
  929. ActLineStartP, ActLineEndP, p, StartPos: PChar;
  930. ExpLine, ActLine: String;
  931. i, LineNo, DiffLineNo: Integer;
  932. begin
  933. writeln('Diff found "',Msg,'". Lines:');
  934. // write correct lines
  935. p:=PChar(Expected);
  936. LineNo:=0;
  937. DiffLineNo:=0;
  938. repeat
  939. StartPos:=p;
  940. while not (p^ in [#0,#10,#13]) do inc(p);
  941. ExpLine:=copy(Expected,StartPos-PChar(Expected)+1,p-StartPos);
  942. SkipLineEnd(p);
  943. inc(LineNo);
  944. if (p<=ExpectedP) and (p^<>#0) then
  945. begin
  946. writeln('= ',ExpLine);
  947. end else begin
  948. // diff line
  949. if DiffLineNo=0 then DiffLineNo:=LineNo;
  950. // write actual line
  951. ActLineStartP:=FindLineStart(ActualP,PChar(Actual));
  952. ActLineEndP:=FindLineEnd(ActualP);
  953. ActLine:=copy(Actual,ActLineStartP-PChar(Actual)+1,ActLineEndP-ActLineStartP);
  954. writeln('- ',ActLine);
  955. // write expected line
  956. writeln('+ ',ExpLine);
  957. // write empty line with pointer ^
  958. for i:=1 to 2+ExpectedP-StartPos do write(' ');
  959. writeln('^');
  960. Msg:='expected "'+ExpLine+'", but got "'+ActLine+'".';
  961. CheckSrcDiff:=false;
  962. // write up to three following actual lines to get some context
  963. for i:=1 to 3 do begin
  964. ActLineStartP:=ActLineEndP;
  965. SkipLineEnd(ActLineStartP);
  966. if ActLineStartP^=#0 then break;
  967. ActLineEndP:=FindLineEnd(ActLineStartP);
  968. ActLine:=copy(Actual,ActLineStartP-PChar(Actual)+1,ActLineEndP-ActLineStartP);
  969. writeln('~ ',ActLine);
  970. end;
  971. exit;
  972. end;
  973. until p^=#0;
  974. writeln('DiffFound Actual:-----------------------');
  975. writeln(Actual);
  976. writeln('DiffFound Expected:---------------------');
  977. writeln(Expected);
  978. writeln('DiffFound ------------------------------');
  979. Msg:='diff found, but lines are the same, internal error';
  980. CheckSrcDiff:=false;
  981. end;
  982. var
  983. IsSpaceNeeded: Boolean;
  984. LastChar, Quote: Char;
  985. begin
  986. Result:=true;
  987. Msg:='';
  988. if Expected='' then Expected:=' ';
  989. if Actual='' then Actual:=' ';
  990. ExpectedP:=PChar(Expected);
  991. ActualP:=PChar(Actual);
  992. repeat
  993. //writeln('TTestModule.CheckDiff Exp="',ExpectedP^,'" Act="',ActualP^,'"');
  994. case ExpectedP^ of
  995. #0:
  996. begin
  997. // check that rest of Actual has only spaces
  998. while ActualP^ in SpaceChars do inc(ActualP);
  999. if ActualP^<>#0 then
  1000. begin
  1001. DiffFound;
  1002. exit;
  1003. end;
  1004. exit(true);
  1005. end;
  1006. ' ',#9,#10,#13:
  1007. begin
  1008. // skip space in Expected
  1009. IsSpaceNeeded:=false;
  1010. if ExpectedP>PChar(Expected) then
  1011. LastChar:=ExpectedP[-1]
  1012. else
  1013. LastChar:=#0;
  1014. while ExpectedP^ in SpaceChars do inc(ExpectedP);
  1015. if (LastChar in ['a'..'z','A'..'Z','0'..'9','_','$'])
  1016. and (ExpectedP^ in ['a'..'z','A'..'Z','0'..'9','_','$']) then
  1017. IsSpaceNeeded:=true;
  1018. if IsSpaceNeeded and (not (ActualP^ in SpaceChars)) then
  1019. begin
  1020. DiffFound;
  1021. exit;
  1022. end;
  1023. while ActualP^ in SpaceChars do inc(ActualP);
  1024. end;
  1025. '''','"':
  1026. begin
  1027. while ActualP^ in SpaceChars do inc(ActualP);
  1028. if ExpectedP^<>ActualP^ then
  1029. begin
  1030. DiffFound;
  1031. exit;
  1032. end;
  1033. Quote:=ExpectedP^;
  1034. repeat
  1035. inc(ExpectedP);
  1036. inc(ActualP);
  1037. if ExpectedP^<>ActualP^ then
  1038. begin
  1039. DiffFound;
  1040. exit;
  1041. end;
  1042. if (ExpectedP^ in [#0,#10,#13]) then
  1043. break
  1044. else if (ExpectedP^=Quote) then
  1045. begin
  1046. inc(ExpectedP);
  1047. inc(ActualP);
  1048. break;
  1049. end;
  1050. until false;
  1051. end;
  1052. else
  1053. while ActualP^ in SpaceChars do inc(ActualP);
  1054. if ExpectedP^<>ActualP^ then
  1055. begin
  1056. DiffFound;
  1057. exit;
  1058. end;
  1059. inc(ExpectedP);
  1060. inc(ActualP);
  1061. end;
  1062. until false;
  1063. end;
  1064. { TTestEnginePasResolver }
  1065. destructor TTestEnginePasResolver.Destroy;
  1066. begin
  1067. FreeAndNil(FStreamResolver);
  1068. FreeAndNil(FParser);
  1069. FreeAndNil(FScanner);
  1070. FreeAndNil(FStreamResolver);
  1071. if Module<>nil then
  1072. begin
  1073. Module.Release{$IFDEF CheckPasTreeRefCount}('CreateElement'){$ENDIF};
  1074. FModule:=nil;
  1075. end;
  1076. inherited Destroy;
  1077. end;
  1078. function TTestEnginePasResolver.FindUnit(const AName, InFilename: String;
  1079. NameExpr, InFileExpr: TPasExpr): TPasModule;
  1080. begin
  1081. Result:=nil;
  1082. if InFilename<>'' then
  1083. RaiseNotYetImplemented(20180224101926,InFileExpr,'Use testcase tcunitsearch instead');
  1084. if Assigned(OnFindUnit) then
  1085. Result:=OnFindUnit(AName);
  1086. if NameExpr=nil then ;
  1087. end;
  1088. procedure TTestEnginePasResolver.UsedInterfacesFinished(Section: TPasSection);
  1089. begin
  1090. // do not parse recursively
  1091. // parse via the queue
  1092. if Section=nil then ;
  1093. end;
  1094. { TCustomTestModule }
  1095. function TCustomTestModule.GetMsgCount: integer;
  1096. begin
  1097. Result:=FHintMsgs.Count;
  1098. end;
  1099. function TCustomTestModule.GetMsgs(Index: integer): TTestHintMessage;
  1100. begin
  1101. Result:=TTestHintMessage(FHintMsgs[Index]);
  1102. end;
  1103. function TCustomTestModule.GetResolverCount: integer;
  1104. begin
  1105. Result:=FModules.Count;
  1106. end;
  1107. function TCustomTestModule.GetResolvers(Index: integer
  1108. ): TTestEnginePasResolver;
  1109. begin
  1110. Result:=TTestEnginePasResolver(FModules[Index]);
  1111. end;
  1112. function TCustomTestModule.OnPasResolverFindUnit(const aUnitName: String
  1113. ): TPasModule;
  1114. var
  1115. DefNamespace: String;
  1116. begin
  1117. //writeln('TTestModule.OnPasResolverFindUnit START Unit="',aUnitName,'"');
  1118. if (Pos('.',aUnitName)<1) then
  1119. begin
  1120. DefNamespace:=GetDefaultNamespace;
  1121. if DefNamespace<>'' then
  1122. begin
  1123. Result:=LoadUnit(DefNamespace+'.'+aUnitName);
  1124. if Result<>nil then exit;
  1125. end;
  1126. end;
  1127. Result:=LoadUnit(aUnitName);
  1128. if Result<>nil then exit;
  1129. {$IFDEF VerbosePas2JS}
  1130. writeln('TTestModule.OnPasResolverFindUnit missing unit "',aUnitName,'"');
  1131. {$ENDIF}
  1132. Fail('can''t find unit "'+aUnitName+'"');
  1133. end;
  1134. procedure TCustomTestModule.OnParserLog(Sender: TObject; const Msg: String);
  1135. var
  1136. aParser: TPasParser;
  1137. Item: TTestHintMessage;
  1138. begin
  1139. aParser:=Sender as TPasParser;
  1140. Item:=TTestHintMessage.Create;
  1141. Item.Id:=aParser.LastMsgNumber;
  1142. Item.MsgType:=aParser.LastMsgType;
  1143. Item.MsgNumber:=aParser.LastMsgNumber;
  1144. Item.Msg:=Msg;
  1145. Item.SourcePos:=aParser.Scanner.CurSourcePos;
  1146. {$IFDEF VerbosePas2JS}
  1147. writeln('TCustomTestModule.OnParserLog ',GetObjName(Sender),' ',Item.MsgType,' (',Item.MsgNumber,') {',Msg,'}');
  1148. {$ENDIF}
  1149. FHintMsgs.Add(Item);
  1150. end;
  1151. procedure TCustomTestModule.OnPasResolverLog(Sender: TObject; const Msg: String
  1152. );
  1153. var
  1154. aResolver: TTestEnginePasResolver;
  1155. Item: TTestHintMessage;
  1156. begin
  1157. aResolver:=Sender as TTestEnginePasResolver;
  1158. Item:=TTestHintMessage.Create;
  1159. Item.Id:=aResolver.LastMsgId;
  1160. Item.MsgType:=aResolver.LastMsgType;
  1161. Item.MsgNumber:=aResolver.LastMsgNumber;
  1162. Item.Msg:=Msg;
  1163. Item.SourcePos:=aResolver.LastSourcePos;
  1164. {$IFDEF VerbosePas2JS}
  1165. writeln('TCustomTestModule.OnPasResolverLog ',GetObjName(Sender),' ',Item.MsgType,' (',Item.MsgNumber,') {',Msg,'}');
  1166. {$ENDIF}
  1167. FHintMsgs.Add(Item);
  1168. end;
  1169. procedure TCustomTestModule.OnScannerLog(Sender: TObject; const Msg: String);
  1170. var
  1171. Item: TTestHintMessage;
  1172. aScanner: TPas2jsPasScanner;
  1173. begin
  1174. aScanner:=Sender as TPas2jsPasScanner;
  1175. Item:=TTestHintMessage.Create;
  1176. Item.Id:=aScanner.LastMsgNumber;
  1177. Item.MsgType:=aScanner.LastMsgType;
  1178. Item.MsgNumber:=aScanner.LastMsgNumber;
  1179. Item.Msg:=Msg;
  1180. Item.SourcePos:=aScanner.CurSourcePos;
  1181. {$IFDEF VerbosePas2JS}
  1182. writeln('TCustomTestModule.OnScannerLog ',GetObjName(Sender),' ',Item.MsgType,' (',Item.MsgNumber,') {',Msg,'}');
  1183. {$ENDIF}
  1184. FHintMsgs.Add(Item);
  1185. end;
  1186. procedure TCustomTestModule.SetWithTypeInfo(const AValue: boolean);
  1187. begin
  1188. if FWithTypeInfo=AValue then Exit;
  1189. FWithTypeInfo:=AValue;
  1190. if AValue then
  1191. Converter.Options:=Converter.Options-[coNoTypeInfo]
  1192. else
  1193. Converter.Options:=Converter.Options+[coNoTypeInfo];
  1194. end;
  1195. function TCustomTestModule.LoadUnit(const aUnitName: String): TPasModule;
  1196. var
  1197. i: Integer;
  1198. CurEngine: TTestEnginePasResolver;
  1199. CurUnitName: String;
  1200. begin
  1201. //writeln('TTestModule.FindUnit START Unit="',aUnitName,'"');
  1202. Result:=nil;
  1203. if (Module.ClassType=TPasModule)
  1204. and (CompareText(Module.Name,aUnitName)=0) then
  1205. exit(Module);
  1206. for i:=0 to ResolverCount-1 do
  1207. begin
  1208. CurEngine:=Resolvers[i];
  1209. CurUnitName:=ExtractFileUnitName(CurEngine.Filename);
  1210. //writeln('TTestModule.FindUnit Checking ',i,'/',ResolverCount,' ',CurEngine.Filename,' ',CurUnitName);
  1211. if CompareText(aUnitName,CurUnitName)=0 then
  1212. begin
  1213. Result:=CurEngine.Module;
  1214. if Result<>nil then exit;
  1215. //writeln('TTestModule.FindUnit PARSING unit "',CurEngine.Filename,'"');
  1216. FileResolver.FindSourceFile(aUnitName);
  1217. CurEngine.StreamResolver:=TStreamResolver.Create;
  1218. CurEngine.StreamResolver.OwnsStreams:=True;
  1219. //writeln('TTestModule.FindUnit SOURCE=',CurEngine.Source);
  1220. CurEngine.StreamResolver.AddStream(CurEngine.FileName,TStringStream.Create(CurEngine.Source));
  1221. CurEngine.Scanner:=TPas2jsPasScanner.Create(CurEngine.StreamResolver);
  1222. InitScanner(CurEngine.Scanner);
  1223. CurEngine.Parser:=TTestPasParser.Create(CurEngine.Scanner,CurEngine.StreamResolver,CurEngine);
  1224. CurEngine.Parser.Options:=po_tcmodules;
  1225. if CompareText(CurUnitName,'System')=0 then
  1226. CurEngine.Parser.ImplicitUses.Clear;
  1227. CurEngine.Scanner.OpenFile(CurEngine.Filename);
  1228. try
  1229. CurEngine.Parser.NextToken;
  1230. CurEngine.Parser.ParseUnit(CurEngine.FModule);
  1231. except
  1232. on E: Exception do
  1233. HandleException(E);
  1234. end;
  1235. //writeln('TTestModule.FindUnit END ',CurUnitName);
  1236. Result:=CurEngine.Module;
  1237. exit;
  1238. end;
  1239. end;
  1240. end;
  1241. procedure TCustomTestModule.SetUp;
  1242. begin
  1243. {$IFDEF EnablePasTreeGlobalRefCount}
  1244. FElementRefCountAtSetup:=TPasElement.GlobalRefCount;
  1245. {$ENDIF}
  1246. if FModules<>nil then
  1247. begin
  1248. writeln('TCustomTestModule.SetUp FModules<>nil');
  1249. Halt;
  1250. end;
  1251. inherited SetUp;
  1252. FSkipTests:=false;
  1253. FWithTypeInfo:=false;
  1254. FSource:=TStringList.Create;
  1255. FHub:=TPas2JSResolverHub.Create(Self);
  1256. FModules:=TObjectList.Create(true);
  1257. FFilename:='test1.pp';
  1258. FFileResolver:=TStreamResolver.Create;
  1259. FFileResolver.OwnsStreams:=True;
  1260. FScanner:=TPas2jsPasScanner.Create(FFileResolver);
  1261. InitScanner(FScanner);
  1262. FEngine:=AddModule(Filename);
  1263. FEngine.Scanner:=FScanner;
  1264. FScanner.Resolver:=FEngine;
  1265. FParser:=TTestPasParser.Create(FScanner,FFileResolver,FEngine);
  1266. FParser.OnLog:=@OnParserLog;
  1267. FEngine.Parser:=FParser;
  1268. Parser.Options:=po_tcmodules;
  1269. FModule:=Nil;
  1270. FConverter:=CreateConverter;
  1271. FExpectedErrorClass:=nil;
  1272. end;
  1273. function TCustomTestModule.CreateConverter: TPasToJSConverter;
  1274. var
  1275. Options: TPasToJsConverterOptions;
  1276. begin
  1277. Result:=TPasToJSConverter.Create;
  1278. Options:=co_tcmodules;
  1279. if WithTypeInfo then
  1280. Exclude(Options,coNoTypeInfo)
  1281. else
  1282. Include(Options,coNoTypeInfo);
  1283. Result.Options:=Options;
  1284. Result.Globals:=TPasToJSConverterGlobals.Create(Result);
  1285. end;
  1286. procedure TCustomTestModule.InitScanner(aScanner: TPas2jsPasScanner);
  1287. begin
  1288. aScanner.AllowedModeSwitches:=msAllPas2jsModeSwitches;
  1289. aScanner.ReadOnlyModeSwitches:=msAllPas2jsModeSwitchesReadOnly;
  1290. aScanner.CurrentModeSwitches:=OBJFPCModeSwitches*msAllPas2jsModeSwitches+msAllPas2jsModeSwitchesReadOnly;
  1291. aScanner.AllowedBoolSwitches:=bsAllPas2jsBoolSwitches;
  1292. aScanner.ReadOnlyBoolSwitches:=bsAllPas2jsBoolSwitchesReadOnly;
  1293. aScanner.CurrentBoolSwitches:=bsAllPas2jsBoolSwitchesReadOnly+[bsHints,bsNotes,bsWarnings,bsWriteableConst];
  1294. aScanner.AllowedValueSwitches:=vsAllPas2jsValueSwitches;
  1295. aScanner.ReadOnlyValueSwitches:=vsAllPas2jsValueSwitchesReadOnly;
  1296. aScanner.OnLog:=@OnScannerLog;
  1297. aScanner.CompilerVersion:='Comp.Ver.tcmodules';
  1298. end;
  1299. procedure TCustomTestModule.TearDown;
  1300. {$IFDEF CheckPasTreeRefCount}
  1301. var
  1302. El: TPasElement;
  1303. {$ENDIF}
  1304. var
  1305. i: Integer;
  1306. CurModule: TPasModule;
  1307. begin
  1308. FHintMsgs.Clear;
  1309. FHintMsgsGood.Clear;
  1310. FSkipTests:=false;
  1311. FWithTypeInfo:=false;
  1312. FJSRegModuleCall:=nil;
  1313. FJSModuleCallArgs:=nil;
  1314. FJSImplentationUses:=nil;
  1315. FJSInterfaceUses:=nil;
  1316. FJSModuleSrc:=nil;
  1317. FJSInitBody:=nil;
  1318. FreeAndNil(FJSSource);
  1319. FreeAndNil(FJSModule);
  1320. FreeAndNil(FConverter);
  1321. Engine.Clear;
  1322. FreeAndNil(FSource);
  1323. FreeAndNil(FFileResolver);
  1324. if FModules<>nil then
  1325. begin
  1326. for i:=0 to FModules.Count-1 do
  1327. begin
  1328. CurModule:=TTestEnginePasResolver(FModules[i]).Module;
  1329. if CurModule=nil then continue;
  1330. //writeln('TCustomTestModule.TearDown ReleaseUsedUnits ',CurModule.Name,' ',CurModule.RefCount,' ',CurModule.RefIds.Text);
  1331. CurModule.ReleaseUsedUnits;
  1332. end;
  1333. if FModule<>nil then
  1334. FModule.ReleaseUsedUnits;
  1335. for i:=0 to FModules.Count-1 do
  1336. begin
  1337. CurModule:=TTestEnginePasResolver(FModules[i]).Module;
  1338. if CurModule=nil then continue;
  1339. //writeln('TCustomTestModule.TearDown UsesReleased ',CurModule.Name,' ',CurModule.RefCount,' ',CurModule.RefIds.Text);
  1340. end;
  1341. FreeAndNil(FModules);
  1342. ReleaseAndNil(TPasElement(FModule){$IFDEF CheckPasTreeRefCount},'CreateElement'{$ENDIF});
  1343. FEngine:=nil;
  1344. end;
  1345. FreeAndNil(FHub);
  1346. inherited TearDown;
  1347. {$IFDEF EnablePasTreeGlobalRefCount}
  1348. if FElementRefCountAtSetup<>TPasElement.GlobalRefCount then
  1349. begin
  1350. writeln('TCustomTestModule.TearDown GlobalRefCount Was='+IntToStr(FElementRefCountAtSetup)+' Now='+IntToStr(TPasElement.GlobalRefCount));
  1351. {$IFDEF CheckPasTreeRefCount}
  1352. El:=TPasElement.FirstRefEl;
  1353. while El<>nil do
  1354. begin
  1355. writeln(' ',GetObjName(El),' RefIds.Count=',El.RefIds.Count,':');
  1356. for i:=0 to El.RefIds.Count-1 do
  1357. writeln(' ',El.RefIds[i]);
  1358. El:=El.NextRefEl;
  1359. end;
  1360. {$ENDIF}
  1361. Halt;
  1362. Fail('TCustomTestModule.TearDown Was='+IntToStr(FElementRefCountAtSetup)+' Now='+IntToStr(TPasElement.GlobalRefCount));
  1363. end;
  1364. {$ENDIF}
  1365. end;
  1366. procedure TCustomTestModule.Add(Line: string);
  1367. begin
  1368. Source.Add(Line);
  1369. end;
  1370. procedure TCustomTestModule.Add(const Lines: array of string);
  1371. var
  1372. i: Integer;
  1373. begin
  1374. for i:=low(Lines) to high(Lines) do
  1375. Add(Lines[i]);
  1376. end;
  1377. procedure TCustomTestModule.StartParsing;
  1378. var
  1379. Src: String;
  1380. begin
  1381. Src:=Source.Text;
  1382. FEngine.Source:=Src;
  1383. FileResolver.AddStream(FileName,TStringStream.Create(Src));
  1384. Scanner.OpenFile(FileName);
  1385. Writeln('// Test : ',Self.TestName);
  1386. Writeln(Src);
  1387. end;
  1388. procedure TCustomTestModule.ParseModuleQueue;
  1389. var
  1390. i: Integer;
  1391. CurResolver: TTestEnginePasResolver;
  1392. Found: Boolean;
  1393. Section: TPasSection;
  1394. begin
  1395. // parse til exception or all modules finished
  1396. while not SkipTests do
  1397. begin
  1398. Found:=false;
  1399. for i:=0 to ResolverCount-1 do
  1400. begin
  1401. CurResolver:=Resolvers[i];
  1402. if CurResolver.CurrentParser=nil then continue;
  1403. if not CurResolver.CurrentParser.CanParseContinue(Section) then
  1404. continue;
  1405. CurResolver.Parser.ParseContinue;
  1406. Found:=true;
  1407. break;
  1408. end;
  1409. if not Found then break;
  1410. end;
  1411. for i:=0 to ResolverCount-1 do
  1412. begin
  1413. CurResolver:=Resolvers[i];
  1414. if CurResolver.Parser=nil then
  1415. begin
  1416. if CurResolver.CurrentParser<>nil then
  1417. Fail('TCustomTestModule.ParseModuleQueue '+CurResolver.Filename+' '+GetObjName(CurResolver.Parser)+'=Parser<>CurrentParser='+GetObjName(CurResolver.CurrentParser));
  1418. continue;
  1419. end;
  1420. if CurResolver.Parser.CurModule<>nil then
  1421. Fail('TCustomTestModule.ParseModuleQueue '+CurResolver.Filename+' NOT FINISHED CurModule='+GetObjName(CurResolver.Parser.CurModule));
  1422. end;
  1423. end;
  1424. procedure TCustomTestModule.ParseModule;
  1425. begin
  1426. if SkipTests then exit;
  1427. FFirstPasStatement:=nil;
  1428. try
  1429. StartParsing;
  1430. Parser.ParseMain(FModule);
  1431. ParseModuleQueue;
  1432. except
  1433. on E: Exception do
  1434. HandleException(E);
  1435. end;
  1436. if SkipTests then exit;
  1437. AssertNotNull('Module resulted in Module',Module);
  1438. AssertEquals('modulename',lowercase(ChangeFileExt(FFileName,'')),lowercase(Module.Name));
  1439. TAssert.AssertSame('Has resolver',Engine,Parser.Engine);
  1440. end;
  1441. procedure TCustomTestModule.ParseProgram;
  1442. begin
  1443. if SkipTests then exit;
  1444. ParseModule;
  1445. if SkipTests then exit;
  1446. AssertEquals('Has program',TPasProgram,Module.ClassType);
  1447. FPasProgram:=TPasProgram(Module);
  1448. AssertNotNull('Has program section',PasProgram.ProgramSection);
  1449. AssertNotNull('Has initialization section',PasProgram.InitializationSection);
  1450. if (PasProgram.InitializationSection.Elements.Count>0) then
  1451. if TObject(PasProgram.InitializationSection.Elements[0]) is TPasImplBlock then
  1452. FFirstPasStatement:=TPasImplBlock(PasProgram.InitializationSection.Elements[0]);
  1453. end;
  1454. procedure TCustomTestModule.ParseUnit;
  1455. begin
  1456. if SkipTests then exit;
  1457. ParseModule;
  1458. if SkipTests then exit;
  1459. AssertEquals('Has unit (TPasModule)',TPasModule,Module.ClassType);
  1460. AssertNotNull('Has interface section',Module.InterfaceSection);
  1461. AssertNotNull('Has implementation section',Module.ImplementationSection);
  1462. if (Module.InitializationSection<>nil)
  1463. and (Module.InitializationSection.Elements.Count>0)
  1464. and (TObject(Module.InitializationSection.Elements[0]) is TPasImplBlock) then
  1465. FFirstPasStatement:=TPasImplBlock(Module.InitializationSection.Elements[0]);
  1466. end;
  1467. function TCustomTestModule.FindModuleWithFilename(aFilename: string
  1468. ): TTestEnginePasResolver;
  1469. var
  1470. i: Integer;
  1471. begin
  1472. for i:=0 to ResolverCount-1 do
  1473. if CompareText(Resolvers[i].Filename,aFilename)=0 then
  1474. exit(Resolvers[i]);
  1475. Result:=nil;
  1476. end;
  1477. function TCustomTestModule.AddModule(aFilename: string
  1478. ): TTestEnginePasResolver;
  1479. begin
  1480. //writeln('TTestModuleConverter.AddModule ',aFilename);
  1481. if FindModuleWithFilename(aFilename)<>nil then
  1482. Fail('TTestModuleConverter.AddModule: file "'+aFilename+'" already exists');
  1483. Result:=TTestEnginePasResolver.Create;
  1484. Result.Filename:=aFilename;
  1485. Result.AddObjFPCBuiltInIdentifiers(btAllJSBaseTypes,bfAllJSBaseProcs);
  1486. Result.OnFindUnit:=@OnPasResolverFindUnit;
  1487. Result.OnLog:=@OnPasResolverLog;
  1488. Result.Hub:=Hub;
  1489. FModules.Add(Result);
  1490. end;
  1491. function TCustomTestModule.AddModuleWithSrc(aFilename, Src: string
  1492. ): TTestEnginePasResolver;
  1493. begin
  1494. Result:=AddModule(aFilename);
  1495. Result.Source:=Src;
  1496. end;
  1497. function TCustomTestModule.AddModuleWithIntfImplSrc(aFilename, InterfaceSrc,
  1498. ImplementationSrc: string): TTestEnginePasResolver;
  1499. var
  1500. Src: String;
  1501. begin
  1502. Src:='unit '+ExtractFileUnitName(aFilename)+';'+LineEnding;
  1503. Src+=LineEnding;
  1504. Src+='interface'+LineEnding;
  1505. Src+=LineEnding;
  1506. Src+=InterfaceSrc;
  1507. Src+='implementation'+LineEnding;
  1508. Src+=LineEnding;
  1509. Src+=ImplementationSrc;
  1510. Src+='end.'+LineEnding;
  1511. Result:=AddModuleWithSrc(aFilename,Src);
  1512. end;
  1513. procedure TCustomTestModule.AddSystemUnit(Parts: TSystemUnitParts);
  1514. var
  1515. Intf, Impl: TStringList;
  1516. begin
  1517. Intf:=TStringList.Create;
  1518. if supTInterfacedObject in Parts then Include(Parts,supTObject);
  1519. // unit interface
  1520. if [supTVarRec,supTypeInfo]*Parts<>[] then
  1521. Intf.Add('{$modeswitch externalclass}');
  1522. Intf.Add('type');
  1523. Intf.Add(' integer=longint;');
  1524. Intf.Add(' sizeint=nativeint;');
  1525. //'const',
  1526. //' LineEnding = #10;',
  1527. //' DirectorySeparator = ''/'';',
  1528. //' DriveSeparator = '''';',
  1529. //' AllowDirectorySeparators : set of char = [''\'',''/''];',
  1530. //' AllowDriveSeparators : set of char = [];',
  1531. if supTObject in Parts then
  1532. Intf.AddStrings([
  1533. 'type',
  1534. ' TClass = class of TObject;',
  1535. ' TObject = class',
  1536. ' constructor Create;',
  1537. ' destructor Destroy; virtual;',
  1538. ' class function ClassType: TClass; assembler;',
  1539. ' class function ClassName: String; assembler;',
  1540. ' class function ClassNameIs(const Name: string): boolean;',
  1541. ' class function ClassParent: TClass; assembler;',
  1542. ' class function InheritsFrom(aClass: TClass): boolean; assembler;',
  1543. ' class function UnitName: String; assembler;',
  1544. ' procedure AfterConstruction; virtual;',
  1545. ' procedure BeforeDestruction;virtual;',
  1546. ' function Equals(Obj: TObject): boolean; virtual;',
  1547. ' function ToString: String; virtual;',
  1548. ' end;']);
  1549. if supTInterfacedObject in Parts then
  1550. Intf.AddStrings([
  1551. ' {$Interfaces COM}',
  1552. ' IUnknown = interface',
  1553. ' [''{00000000-0000-0000-C000-000000000046}'']',
  1554. //' function QueryInterface(const iid: TGuid; out obj): Integer;',
  1555. ' function _AddRef: Integer;',
  1556. ' function _Release: Integer;',
  1557. ' end;',
  1558. ' IInterface = IUnknown;',
  1559. ' TInterfacedObject = class(TObject,IUnknown)',
  1560. ' protected',
  1561. ' fRefCount: Integer;',
  1562. ' { implement methods of IUnknown }',
  1563. //' function QueryInterface(const iid: TGuid; out obj): Integer; virtual;',
  1564. ' function _AddRef: Integer; virtual;',
  1565. ' function _Release: Integer; virtual;',
  1566. ' end;',
  1567. ' TInterfacedClass = class of TInterfacedObject;',
  1568. '',
  1569. '']);
  1570. if supTVarRec in Parts then
  1571. Intf.AddStrings([
  1572. 'const',
  1573. ' vtInteger = 0;',
  1574. ' vtBoolean = 1;',
  1575. ' vtJSValue = 19;',
  1576. 'type',
  1577. ' PVarRec = ^TVarRec;',
  1578. ' TVarRec = record',
  1579. ' VType : byte;',
  1580. ' VJSValue: JSValue;',
  1581. ' vInteger: longint external name ''VJSValue'';',
  1582. ' vBoolean: boolean external name ''VJSValue'';',
  1583. ' end;',
  1584. ' TVarRecArray = array of TVarRec;',
  1585. 'function VarRecs: TVarRecArray; varargs;',
  1586. '']);
  1587. if supTypeInfo in Parts then
  1588. begin
  1589. Intf.AddStrings([
  1590. 'type',
  1591. ' TTypeKind = (',
  1592. ' tkUnknown, // 0',
  1593. ' tkInteger, // 1',
  1594. ' tkChar, // 2 in Delphi/FPC tkWChar, tkUChar',
  1595. ' tkString, // 3 in Delphi/FPC tkSString, tkWString or tkUString',
  1596. ' tkEnumeration, // 4',
  1597. ' tkSet, // 5',
  1598. ' tkDouble, // 6',
  1599. ' tkBool, // 7',
  1600. ' tkProcVar, // 8 function or procedure',
  1601. ' tkMethod, // 9 proc var of object',
  1602. ' tkArray, // 10 static array',
  1603. ' tkDynArray, // 11',
  1604. ' tkRecord, // 12',
  1605. ' tkClass, // 13',
  1606. ' tkClassRef, // 14',
  1607. ' tkPointer, // 15',
  1608. ' tkJSValue, // 16',
  1609. ' tkRefToProcVar, // 17 variable of procedure type',
  1610. ' tkInterface, // 18',
  1611. ' //tkObject,',
  1612. ' //tkSString,tkLString,tkAString,tkWString,',
  1613. ' //tkVariant,',
  1614. ' //tkWChar,',
  1615. ' //tkInt64,',
  1616. ' //tkQWord,',
  1617. ' //tkInterfaceRaw,',
  1618. ' //tkUString,tkUChar,',
  1619. ' tkHelper, // 19',
  1620. ' //tkFile,',
  1621. ' tkExtClass // 20',
  1622. ' );',
  1623. ' TTypeKinds = set of TTypeKind;',
  1624. ' TTypeInfo = class external name ''rtl.tTypeInfo'' end;',
  1625. ' TTypeInfoInteger = class external name ''rtl.tTypeInfoInteger''(TTypeInfo)',
  1626. ' end;',
  1627. ' TTypeInfoEnum = class external name ''rtl.tTypeInfoEnum''(TTypeInfoInteger) end;',
  1628. ' TTypeInfoSet = class external name ''rtl.tTypeInfoSet''(TTypeInfo) end;',
  1629. ' TTypeInfoStaticArray = class external name ''rtl.tTypeInfoStaticArray''(TTypeInfo) end;',
  1630. ' TTypeInfoDynArray = class external name ''rtl.tTypeInfoDynArray''(TTypeInfo) end;',
  1631. ' TTypeInfoProcVar = class external name ''rtl.tTypeInfoProcVar''(TTypeInfo) end;',
  1632. ' TTypeInfoMethodVar = class external name ''rtl.tTypeInfoMethodVar''(TTypeInfoProcVar) end;',
  1633. ' TTypeInfoClass = class external name ''rtl.tTypeInfoClass''(TTypeInfo) end;',
  1634. ' TTypeInfoClassRef = class external name ''rtl.tTypeInfoClassRef''(TTypeInfo) end;',
  1635. ' TTypeInfoExtClass = class external name ''rtl.tTypeInfoExtClass''(TTypeInfo) end;',
  1636. ' TTypeInfoRecord = class external name ''rtl.tTypeInfoRecord''(TTypeInfo) end;',
  1637. ' TTypeInfoPointer = class external name ''rtl.tTypeInfoPointer''(TTypeInfo) end;',
  1638. ' TTypeInfoHelper = class external name ''rtl.tTypeInfoHelper''(TTypeInfo) end;',
  1639. ' TTypeInfoInterface = class external name ''rtl.tTypeInfoInterface''(TTypeInfo) end;',
  1640. '']);
  1641. end;
  1642. Intf.Add('var');
  1643. Intf.Add(' ExitCode: Longint = 0;');
  1644. // unit implementation
  1645. Impl:=TStringList.Create;
  1646. if supTObject in Parts then
  1647. Impl.AddStrings([
  1648. '// needed by ClassNameIs, the real SameText is in SysUtils',
  1649. 'function SameText(const s1, s2: String): Boolean; assembler;',
  1650. 'asm',
  1651. 'end;',
  1652. 'constructor TObject.Create; begin end;',
  1653. 'destructor TObject.Destroy; begin end;',
  1654. 'class function TObject.ClassType: TClass; assembler;',
  1655. 'asm',
  1656. 'end;',
  1657. 'class function TObject.ClassName: String; assembler;',
  1658. 'asm',
  1659. 'end;',
  1660. 'class function TObject.ClassNameIs(const Name: string): boolean;',
  1661. 'begin',
  1662. ' Result:=SameText(Name,ClassName);',
  1663. 'end;',
  1664. 'class function TObject.ClassParent: TClass; assembler;',
  1665. 'asm',
  1666. 'end;',
  1667. 'class function TObject.InheritsFrom(aClass: TClass): boolean; assembler;',
  1668. 'asm',
  1669. 'end;',
  1670. 'class function TObject.UnitName: String; assembler;',
  1671. 'asm',
  1672. 'end;',
  1673. 'procedure TObject.AfterConstruction; begin end;',
  1674. 'procedure TObject.BeforeDestruction; begin end;',
  1675. 'function TObject.Equals(Obj: TObject): boolean;',
  1676. 'begin',
  1677. ' Result:=Obj=Self;',
  1678. 'end;',
  1679. 'function TObject.ToString: String;',
  1680. 'begin',
  1681. ' Result:=ClassName;',
  1682. 'end;'
  1683. ]);
  1684. if supTInterfacedObject in Parts then
  1685. Impl.AddStrings([
  1686. //'function TInterfacedObject.QueryInterface(const iid: TGuid; out obj): Integer;',
  1687. //'begin',
  1688. //'end;',
  1689. 'function TInterfacedObject._AddRef: Integer;',
  1690. 'begin',
  1691. 'end;',
  1692. 'function TInterfacedObject._Release: Integer;',
  1693. 'begin',
  1694. 'end;',
  1695. '']);
  1696. if supTVarRec in Parts then
  1697. Impl.AddStrings([
  1698. 'function VarRecs: TVarRecArray; varargs;',
  1699. 'var',
  1700. ' v: PVarRec;',
  1701. 'begin',
  1702. ' v^.VType:=1;',
  1703. ' v^.VJSValue:=2;',
  1704. 'end;',
  1705. '']);
  1706. try
  1707. AddModuleWithIntfImplSrc('system.pp',Intf.Text,Impl.Text);
  1708. finally
  1709. Intf.Free;
  1710. Impl.Free;
  1711. end;
  1712. end;
  1713. procedure TCustomTestModule.StartProgram(NeedSystemUnit: boolean;
  1714. SystemUnitParts: TSystemUnitParts);
  1715. begin
  1716. if NeedSystemUnit then
  1717. AddSystemUnit(SystemUnitParts)
  1718. else
  1719. Parser.ImplicitUses.Clear;
  1720. Add('program '+ExtractFileUnitName(Filename)+';');
  1721. Add('');
  1722. end;
  1723. procedure TCustomTestModule.StartUnit(NeedSystemUnit: boolean;
  1724. SystemUnitParts: TSystemUnitParts);
  1725. begin
  1726. if NeedSystemUnit then
  1727. AddSystemUnit(SystemUnitParts)
  1728. else
  1729. Parser.ImplicitUses.Clear;
  1730. Add('unit Test1;');
  1731. Add('');
  1732. end;
  1733. procedure TCustomTestModule.ConvertModule;
  1734. procedure CheckUsesList(UsesName: String; Arg: TJSArrayLiteralElement;
  1735. out UsesLit: TJSArrayLiteral);
  1736. var
  1737. i: Integer;
  1738. Item: TJSElement;
  1739. Lit: TJSLiteral;
  1740. begin
  1741. UsesLit:=nil;
  1742. AssertNotNull(UsesName+' uses section',Arg.Expr);
  1743. if (Arg.Expr.ClassType=TJSLiteral) and TJSLiteral(Arg.Expr).Value.IsNull then
  1744. exit; // null is ok
  1745. AssertEquals(UsesName+' uses section param is array',TJSArrayLiteral,Arg.Expr.ClassType);
  1746. FJSInterfaceUses:=TJSArrayLiteral(Arg.Expr);
  1747. for i:=0 to FJSInterfaceUses.Elements.Count-1 do
  1748. begin
  1749. Item:=FJSInterfaceUses.Elements.Elements[i].Expr;
  1750. AssertNotNull(UsesName+' uses section item['+IntToStr(i)+'].Expr',Item);
  1751. AssertEquals(UsesName+' uses section item['+IntToStr(i)+'] is lit',TJSLiteral,Item.ClassType);
  1752. Lit:=TJSLiteral(Item);
  1753. AssertEquals(UsesName+' uses section item['+IntToStr(i)+'] is string lit',
  1754. ord(jsbase.jstString),ord(Lit.Value.ValueType));
  1755. end;
  1756. end;
  1757. procedure CheckFunctionParam(ParamName: string; Arg: TJSArrayLiteralElement;
  1758. out Src: TJSSourceElements);
  1759. var
  1760. FunDecl: TJSFunctionDeclarationStatement;
  1761. FunDef: TJSFuncDef;
  1762. FunBody: TJSFunctionBody;
  1763. begin
  1764. Src:=nil;
  1765. AssertNotNull(ParamName,Arg.Expr);
  1766. AssertEquals(ParamName+' Arg.Expr type',TJSFunctionDeclarationStatement,Arg.Expr.ClassType);
  1767. FunDecl:=Arg.Expr as TJSFunctionDeclarationStatement;
  1768. AssertNotNull(ParamName+' FunDecl.AFunction',FunDecl.AFunction);
  1769. AssertEquals(ParamName+' FunDecl.AFunction type',TJSFuncDef,FunDecl.AFunction.ClassType);
  1770. FunDef:=FunDecl.AFunction as TJSFuncDef;
  1771. AssertEquals(ParamName+' name empty','',String(FunDef.Name));
  1772. AssertNotNull(ParamName+' body',FunDef.Body);
  1773. AssertEquals(ParamName+' body type',TJSFunctionBody,FunDef.Body.ClassType);
  1774. FunBody:=FunDef.Body as TJSFunctionBody;
  1775. AssertNotNull(ParamName+' body.A',FunBody.A);
  1776. AssertEquals(ParamName+' body.A type',TJSSourceElements,FunBody.A.ClassType);
  1777. Src:=FunBody.A as TJSSourceElements;
  1778. end;
  1779. var
  1780. ModuleNameExpr: TJSLiteral;
  1781. InitFunction: TJSFunctionDeclarationStatement;
  1782. InitAssign: TJSSimpleAssignStatement;
  1783. InitName: String;
  1784. LastNode: TJSElement;
  1785. Arg: TJSArrayLiteralElement;
  1786. begin
  1787. if SkipTests then exit;
  1788. try
  1789. FJSModule:=FConverter.ConvertPasElement(Module,Engine) as TJSSourceElements;
  1790. except
  1791. on E: Exception do
  1792. HandleException(E);
  1793. end;
  1794. if SkipTests then exit;
  1795. if ExpectedErrorClass<>nil then
  1796. Fail('Missing '+ExpectedErrorClass.ClassName+' error {'+ExpectedErrorMsg+'} ('+IntToStr(ExpectedErrorNumber)+')');
  1797. FJSSource:=TStringList.Create;
  1798. FJSSource.Text:=ConvertJSModuleToString(JSModule);
  1799. {$IFDEF VerbosePas2JS}
  1800. writeln('TTestModule.ConvertModule JS:');
  1801. write(FJSSource.Text);
  1802. {$ENDIF}
  1803. // rtl.module(...
  1804. AssertEquals('jsmodule has one statement - the call',1,JSModule.Statements.Count);
  1805. AssertNotNull('register module call',JSModule.Statements.Nodes[0].Node);
  1806. AssertEquals('register module call',TJSCallExpression,JSModule.Statements.Nodes[0].Node.ClassType);
  1807. FJSRegModuleCall:=JSModule.Statements.Nodes[0].Node as TJSCallExpression;
  1808. AssertNotNull('register module rtl.module expr',JSRegModuleCall.Expr);
  1809. AssertNotNull('register module rtl.module args',JSRegModuleCall.Args);
  1810. AssertEquals('rtl.module args',TJSArguments,JSRegModuleCall.Args.ClassType);
  1811. FJSModuleCallArgs:=JSRegModuleCall.Args as TJSArguments;
  1812. // parameter 'unitname'
  1813. if JSModuleCallArgs.Elements.Count<1 then
  1814. Fail('rtl.module first param unit missing');
  1815. Arg:=JSModuleCallArgs.Elements.Elements[0];
  1816. AssertNotNull('module name param',Arg.Expr);
  1817. ModuleNameExpr:=Arg.Expr as TJSLiteral;
  1818. AssertEquals('module name param is string',ord(jstString),ord(ModuleNameExpr.Value.ValueType));
  1819. if Module is TPasProgram then
  1820. AssertEquals('module name','program',String(ModuleNameExpr.Value.AsString))
  1821. else
  1822. AssertEquals('module name',Module.Name,String(ModuleNameExpr.Value.AsString));
  1823. // main uses section
  1824. if JSModuleCallArgs.Elements.Count<2 then
  1825. Fail('rtl.module second param main uses missing');
  1826. Arg:=JSModuleCallArgs.Elements.Elements[1];
  1827. CheckUsesList('interface',Arg,FJSInterfaceUses);
  1828. // program/library/interface function()
  1829. if JSModuleCallArgs.Elements.Count<3 then
  1830. Fail('rtl.module third param intf-function missing');
  1831. Arg:=JSModuleCallArgs.Elements.Elements[2];
  1832. CheckFunctionParam('module intf-function',Arg,FJSModuleSrc);
  1833. // search for $mod.$init or $mod.$main - the last statement
  1834. if Module is TPasProgram then
  1835. begin
  1836. InitName:='$main';
  1837. AssertEquals('$mod.'+InitName+' function 1',true,JSModuleSrc.Statements.Count>0);
  1838. end
  1839. else
  1840. InitName:='$init';
  1841. FJSInitBody:=nil;
  1842. if JSModuleSrc.Statements.Count>0 then
  1843. begin
  1844. LastNode:=JSModuleSrc.Statements.Nodes[JSModuleSrc.Statements.Count-1].Node;
  1845. if LastNode is TJSSimpleAssignStatement then
  1846. begin
  1847. InitAssign:=LastNode as TJSSimpleAssignStatement;
  1848. if GetDottedIdentifier(InitAssign.LHS)='$mod.'+InitName then
  1849. begin
  1850. InitFunction:=InitAssign.Expr as TJSFunctionDeclarationStatement;
  1851. FJSInitBody:=InitFunction.AFunction.Body as TJSFunctionBody;
  1852. end
  1853. else if Module is TPasProgram then
  1854. CheckDottedIdentifier('init function',InitAssign.LHS,'$mod.'+InitName);
  1855. end;
  1856. end;
  1857. // optional: implementation uses section
  1858. if JSModuleCallArgs.Elements.Count<4 then
  1859. exit;
  1860. Arg:=JSModuleCallArgs.Elements.Elements[3];
  1861. CheckUsesList('implementation',Arg,FJSImplentationUses);
  1862. end;
  1863. procedure TCustomTestModule.ConvertProgram;
  1864. begin
  1865. Add('end.');
  1866. ParseProgram;
  1867. ConvertModule;
  1868. end;
  1869. procedure TCustomTestModule.ConvertUnit;
  1870. begin
  1871. Add('end.');
  1872. ParseUnit;
  1873. ConvertModule;
  1874. end;
  1875. function TCustomTestModule.ConvertJSModuleToString(El: TJSElement): string;
  1876. begin
  1877. Result:=tcmodules.JSToStr(El);
  1878. end;
  1879. procedure TCustomTestModule.CheckDottedIdentifier(Msg: string; El: TJSElement;
  1880. DottedName: string);
  1881. begin
  1882. if DottedName='' then
  1883. begin
  1884. AssertNull(Msg,El);
  1885. end
  1886. else
  1887. begin
  1888. AssertNotNull(Msg,El);
  1889. AssertEquals(Msg,DottedName,GetDottedIdentifier(El));
  1890. end;
  1891. end;
  1892. function TCustomTestModule.GetDottedIdentifier(El: TJSElement): string;
  1893. begin
  1894. if El=nil then
  1895. Result:=''
  1896. else if El is TJSPrimaryExpressionIdent then
  1897. Result:=String(TJSPrimaryExpressionIdent(El).Name)
  1898. else if El is TJSDotMemberExpression then
  1899. Result:=GetDottedIdentifier(TJSDotMemberExpression(El).MExpr)+'.'+String(TJSDotMemberExpression(El).Name)
  1900. else
  1901. AssertEquals('GetDottedIdentifier',TJSPrimaryExpressionIdent,El.ClassType);
  1902. end;
  1903. procedure TCustomTestModule.CheckSource(Msg, Statements: String;
  1904. InitStatements: string; ImplStatements: string);
  1905. var
  1906. ActualSrc, ExpectedSrc, InitName: String;
  1907. begin
  1908. ActualSrc:=JSToStr(JSModuleSrc);
  1909. if coUseStrict in Converter.Options then
  1910. ExpectedSrc:='"use strict";'+LineEnding
  1911. else
  1912. ExpectedSrc:='';
  1913. ExpectedSrc:=ExpectedSrc+'var $mod = this;'+LineEnding;
  1914. ExpectedSrc:=ExpectedSrc+Statements;
  1915. // unit implementation
  1916. if (Trim(ImplStatements)<>'') then
  1917. ExpectedSrc:=ExpectedSrc+LineEnding
  1918. +'$mod.$implcode = function () {'+LineEnding
  1919. +ImplStatements
  1920. +'};'+LineEnding;
  1921. // program main or unit initialization
  1922. if (Module is TPasProgram) or (Trim(InitStatements)<>'') then
  1923. begin
  1924. if Module is TPasProgram then
  1925. InitName:='$main'
  1926. else
  1927. InitName:='$init';
  1928. ExpectedSrc:=ExpectedSrc+LineEnding
  1929. +'$mod.'+InitName+' = function () {'+LineEnding
  1930. +InitStatements
  1931. +'};'+LineEnding;
  1932. end;
  1933. //writeln('TCustomTestModule.CheckSource ExpectedIntf="',ExpectedSrc,'"');
  1934. //writeln('TTestModule.CheckSource InitStatements="',Trim(InitStatements),'"');
  1935. CheckDiff(Msg,ExpectedSrc,ActualSrc);
  1936. end;
  1937. procedure TCustomTestModule.CheckDiff(Msg, Expected, Actual: string);
  1938. // search diff, ignore changes in spaces
  1939. var
  1940. s: string;
  1941. begin
  1942. if CheckSrcDiff(Expected,Actual,s) then exit;
  1943. Fail(Msg+': '+s);
  1944. end;
  1945. procedure TCustomTestModule.CheckUnit(Filename, ExpectedSrc: string);
  1946. var
  1947. aResolver: TTestEnginePasResolver;
  1948. aConverter: TPasToJSConverter;
  1949. aJSModule: TJSSourceElements;
  1950. ActualSrc: String;
  1951. begin
  1952. aResolver:=GetResolver(Filename);
  1953. AssertNotNull('missing resolver of unit '+Filename,aResolver);
  1954. AssertNotNull('missing resolver.module of unit '+Filename,aResolver.Module);
  1955. {$IFDEF VerbosePas2JS}
  1956. writeln('CheckUnit '+Filename+' converting ...');
  1957. {$ENDIF}
  1958. aConverter:=CreateConverter;
  1959. aJSModule:=nil;
  1960. try
  1961. try
  1962. aJSModule:=aConverter.ConvertPasElement(aResolver.Module,aResolver) as TJSSourceElements;
  1963. except
  1964. on E: Exception do
  1965. HandleException(E);
  1966. end;
  1967. ActualSrc:=ConvertJSModuleToString(aJSModule);
  1968. {$IFDEF VerbosePas2JS}
  1969. writeln('TTestModule.CheckUnit ',Filename,' Pas:');
  1970. write(aResolver.Source);
  1971. writeln('TTestModule.CheckUnit ',Filename,' JS:');
  1972. write(ActualSrc);
  1973. {$ENDIF}
  1974. CheckDiff('Converted unit: "'+ChangeFileExt(Filename,'.js')+'"',ExpectedSrc,ActualSrc);
  1975. finally
  1976. aJSModule.Free;
  1977. aConverter.Free;
  1978. end;
  1979. end;
  1980. procedure TCustomTestModule.CheckHint(MsgType: TMessageType;
  1981. MsgNumber: integer; Msg: string; Marker: PSrcMarker);
  1982. var
  1983. i: Integer;
  1984. Item: TTestHintMessage;
  1985. Expected,Actual: string;
  1986. begin
  1987. //writeln('TCustomTestModule.CheckHint MsgCount=',MsgCount);
  1988. for i:=0 to MsgCount-1 do
  1989. begin
  1990. Item:=Msgs[i];
  1991. if (Item.MsgNumber<>MsgNumber) or (Item.Msg<>Msg) then continue;
  1992. if (Marker<>nil) then
  1993. begin
  1994. if Item.SourcePos.Row<>cardinal(Marker^.Row) then continue;
  1995. if (Item.SourcePos.Column<cardinal(Marker^.StartCol))
  1996. or (Item.SourcePos.Column>cardinal(Marker^.EndCol)) then continue;
  1997. end;
  1998. // found
  1999. FHintMsgsGood.Add(Item);
  2000. str(Item.MsgType,Actual);
  2001. str(MsgType,Expected);
  2002. AssertEquals('MsgType',Expected,Actual);
  2003. exit;
  2004. end;
  2005. // needed message missing -> show emitted messages
  2006. WriteSources('',0,0);
  2007. for i:=0 to MsgCount-1 do
  2008. begin
  2009. Item:=Msgs[i];
  2010. write('TCustomTestModule.CheckHint ',i,'/',MsgCount,' ',Item.MsgType,
  2011. ' ('+IntToStr(Item.MsgNumber),')');
  2012. if Marker<>nil then
  2013. write(' '+ExtractFileName(Item.SourcePos.FileName),'(',Item.SourcePos.Row,',',Item.SourcePos.Column,')');
  2014. writeln(' {',Item.Msg,'}');
  2015. end;
  2016. str(MsgType,Expected);
  2017. Actual:='Missing '+Expected+' ('+IntToStr(MsgNumber)+')';
  2018. if Marker<>nil then
  2019. Actual:=Actual+' '+ExtractFileName(Marker^.Filename)+'('+IntToStr(Marker^.Row)+','+IntToStr(Marker^.StartCol)+'..'+IntToStr(Marker^.EndCol)+')';
  2020. Actual:=Actual+' '+Msg;
  2021. Fail(Actual);
  2022. end;
  2023. procedure TCustomTestModule.CheckResolverUnexpectedHints(WithSourcePos: boolean
  2024. );
  2025. var
  2026. i: Integer;
  2027. s, Txt: String;
  2028. Msg: TTestHintMessage;
  2029. begin
  2030. for i:=0 to MsgCount-1 do
  2031. begin
  2032. Msg:=Msgs[i];
  2033. if FHintMsgsGood.IndexOf(Msg)>=0 then continue;
  2034. s:='';
  2035. str(Msg.MsgType,s);
  2036. Txt:='Unexpected resolver message found ['+IntToStr(Msg.Id)+'] '
  2037. +s+': ('+IntToStr(Msg.MsgNumber)+')';
  2038. if WithSourcePos then
  2039. Txt:=Txt+' '+ExtractFileName(Msg.SourcePos.FileName)+'('+IntToStr(Msg.SourcePos.Row)+','+IntToStr(Msg.SourcePos.Column)+')';
  2040. Txt:=Txt+' {'+Msg.Msg+'}';
  2041. Fail(Txt);
  2042. end;
  2043. end;
  2044. procedure TCustomTestModule.SetExpectedScannerError(Msg: string;
  2045. MsgNumber: integer);
  2046. begin
  2047. ExpectedErrorClass:=EScannerError;
  2048. ExpectedErrorMsg:=Msg;
  2049. ExpectedErrorNumber:=MsgNumber;
  2050. end;
  2051. procedure TCustomTestModule.SetExpectedParserError(Msg: string;
  2052. MsgNumber: integer);
  2053. begin
  2054. ExpectedErrorClass:=EParserError;
  2055. ExpectedErrorMsg:=Msg;
  2056. ExpectedErrorNumber:=MsgNumber;
  2057. end;
  2058. procedure TCustomTestModule.SetExpectedPasResolverError(Msg: string;
  2059. MsgNumber: integer);
  2060. begin
  2061. ExpectedErrorClass:=EPasResolve;
  2062. ExpectedErrorMsg:=Msg;
  2063. ExpectedErrorNumber:=MsgNumber;
  2064. end;
  2065. procedure TCustomTestModule.SetExpectedConverterError(Msg: string;
  2066. MsgNumber: integer);
  2067. begin
  2068. ExpectedErrorClass:=EPas2JS;
  2069. ExpectedErrorMsg:=Msg;
  2070. ExpectedErrorNumber:=MsgNumber;
  2071. end;
  2072. function TCustomTestModule.IsErrorExpected(E: Exception): boolean;
  2073. var
  2074. MsgNumber: Integer;
  2075. Msg: String;
  2076. begin
  2077. Result:=false;
  2078. if (ExpectedErrorClass=nil) or (ExpectedErrorClass<>E.ClassType) then exit;
  2079. Msg:=E.Message;
  2080. if E is EPas2JS then
  2081. MsgNumber:=EPas2JS(E).MsgNumber
  2082. else if E is EPasResolve then
  2083. MsgNumber:=EPasResolve(E).MsgNumber
  2084. else if E is EParserError then
  2085. MsgNumber:=Parser.LastMsgNumber
  2086. else if E is EScannerError then
  2087. begin
  2088. MsgNumber:=Scanner.LastMsgNumber;
  2089. Msg:=Scanner.LastMsg;
  2090. end
  2091. else
  2092. MsgNumber:=0;
  2093. Result:=(MsgNumber=ExpectedErrorNumber) and (Msg=ExpectedErrorMsg);
  2094. if Result then
  2095. SkipTests:=true;
  2096. end;
  2097. procedure TCustomTestModule.HandleScannerError(E: EScannerError);
  2098. begin
  2099. if IsErrorExpected(E) then exit;
  2100. WriteSources(Scanner.CurFilename,Scanner.CurRow,Scanner.CurColumn);
  2101. writeln('ERROR: TCustomTestModule.HandleScannerError '+E.ClassName+':'+E.Message
  2102. +' '+Scanner.CurFilename
  2103. +'('+IntToStr(Scanner.CurRow)+','+IntToStr(Scanner.CurColumn)+')');
  2104. FailException(E);
  2105. end;
  2106. procedure TCustomTestModule.HandleParserError(E: EParserError);
  2107. begin
  2108. if IsErrorExpected(E) then exit;
  2109. WriteSources(E.Filename,E.Row,E.Column);
  2110. writeln('ERROR: TCustomTestModule.HandleParserError '+E.ClassName+':'+E.Message
  2111. +' '+E.Filename+'('+IntToStr(E.Row)+','+IntToStr(E.Column)+')'
  2112. +' MainModuleScannerLine="'+Scanner.CurLine+'"'
  2113. );
  2114. FailException(E);
  2115. end;
  2116. procedure TCustomTestModule.HandlePasResolveError(E: EPasResolve);
  2117. var
  2118. P: TPasSourcePos;
  2119. begin
  2120. if IsErrorExpected(E) then exit;
  2121. P:=E.SourcePos;
  2122. WriteSources(P.FileName,P.Row,P.Column);
  2123. writeln('ERROR: TCustomTestModule.HandlePasResolveError '+E.ClassName+':'+E.Message
  2124. +' '+P.FileName+'('+IntToStr(P.Row)+','+IntToStr(P.Column)+')');
  2125. FailException(E);
  2126. end;
  2127. procedure TCustomTestModule.HandlePas2JSError(E: EPas2JS);
  2128. var
  2129. Row, Col: integer;
  2130. begin
  2131. if IsErrorExpected(E) then exit;
  2132. Engine.UnmangleSourceLineNumber(E.PasElement.SourceLinenumber,Row,Col);
  2133. WriteSources(E.PasElement.SourceFilename,Row,Col);
  2134. writeln('ERROR: TCustomTestModule.HandlePas2JSError '+E.ClassName+':'+E.Message
  2135. +' '+E.PasElement.SourceFilename
  2136. +'('+IntToStr(Row)+','+IntToStr(Col)+')');
  2137. FailException(E);
  2138. end;
  2139. procedure TCustomTestModule.HandleException(E: Exception);
  2140. begin
  2141. if E is EScannerError then
  2142. HandleScannerError(EScannerError(E))
  2143. else if E is EParserError then
  2144. HandleParserError(EParserError(E))
  2145. else if E is EPasResolve then
  2146. HandlePasResolveError(EPasResolve(E))
  2147. else if E is EPas2JS then
  2148. HandlePas2JSError(EPas2JS(E))
  2149. else
  2150. begin
  2151. if IsErrorExpected(E) then exit;
  2152. if not (E is EAssertionFailedError) then
  2153. begin
  2154. WriteSources('',0,0);
  2155. writeln('ERROR: TCustomTestModule.HandleException '+E.ClassName+':'+E.Message);
  2156. end;
  2157. FailException(E);
  2158. end;
  2159. end;
  2160. procedure TCustomTestModule.FailException(E: Exception);
  2161. var
  2162. MsgNumber: Integer;
  2163. begin
  2164. if ExpectedErrorClass<>nil then
  2165. begin
  2166. if FExpectedErrorClass=E.ClassType then
  2167. begin
  2168. if E is EPas2JS then
  2169. MsgNumber:=EPas2JS(E).MsgNumber
  2170. else if E is EPasResolve then
  2171. MsgNumber:=EPasResolve(E).MsgNumber
  2172. else if E is EParserError then
  2173. MsgNumber:=Parser.LastMsgNumber
  2174. else if E is EScannerError then
  2175. MsgNumber:=Scanner.LastMsgNumber
  2176. else
  2177. MsgNumber:=0;
  2178. AssertEquals('Expected error message ('+IntToStr(ExpectedErrorNumber)+')','{'+ExpectedErrorMsg+'}','{'+E.Message+'}');
  2179. AssertEquals('Expected {'+ExpectedErrorMsg+'}, but got msg {'+E.Message+'} number',
  2180. ExpectedErrorNumber,MsgNumber);
  2181. end else begin
  2182. AssertEquals('Wrong exception class',ExpectedErrorClass.ClassName,E.ClassName);
  2183. end;
  2184. end;
  2185. Fail(E.Message);
  2186. end;
  2187. procedure TCustomTestModule.WriteSources(const aFilename: string; aRow,
  2188. aCol: integer);
  2189. var
  2190. IsSrc: Boolean;
  2191. i, j: Integer;
  2192. SrcLines: TStringList;
  2193. Line: string;
  2194. aModule: TTestEnginePasResolver;
  2195. begin
  2196. writeln('TCustomTestModule.WriteSources File="',aFilename,'" Row=',aRow,' Col=',aCol);
  2197. for i:=0 to ResolverCount-1 do
  2198. begin
  2199. aModule:=Resolvers[i];
  2200. SrcLines:=TStringList.Create;
  2201. try
  2202. SrcLines.Text:=aModule.Source;
  2203. IsSrc:=ExtractFilename(aModule.Filename)=ExtractFileName(aFilename);
  2204. writeln('Testcode:-File="',aModule.Filename,'"----------------------------------:');
  2205. for j:=1 to SrcLines.Count do
  2206. begin
  2207. Line:=SrcLines[j-1];
  2208. if IsSrc and (j=aRow) then
  2209. begin
  2210. write('*');
  2211. Line:=LeftStr(Line,aCol-1)+'|'+copy(Line,aCol,length(Line));
  2212. end;
  2213. writeln(Format('%:4d: ',[j]),Line);
  2214. end;
  2215. finally
  2216. SrcLines.Free;
  2217. end;
  2218. end;
  2219. end;
  2220. function TCustomTestModule.IndexOfResolver(const Filename: string): integer;
  2221. var
  2222. i: Integer;
  2223. begin
  2224. for i:=0 to ResolverCount-1 do
  2225. if Filename=Resolvers[i].Filename then exit(i);
  2226. Result:=-1;
  2227. end;
  2228. function TCustomTestModule.GetResolver(const Filename: string
  2229. ): TTestEnginePasResolver;
  2230. var
  2231. i: Integer;
  2232. begin
  2233. i:=IndexOfResolver(Filename);
  2234. if i<0 then exit(nil);
  2235. Result:=Resolvers[i];
  2236. end;
  2237. function TCustomTestModule.GetDefaultNamespace: string;
  2238. var
  2239. C: TClass;
  2240. begin
  2241. Result:='';
  2242. if FModule=nil then exit;
  2243. C:=FModule.ClassType;
  2244. if (C=TPasProgram) or (C=TPasLibrary) or (C=TPasPackage) then
  2245. Result:=Engine.DefaultNameSpace;
  2246. end;
  2247. constructor TCustomTestModule.Create;
  2248. begin
  2249. inherited Create;
  2250. FHintMsgs:=TObjectList.Create(true);
  2251. FHintMsgsGood:=TFPList.Create;
  2252. end;
  2253. destructor TCustomTestModule.Destroy;
  2254. begin
  2255. FreeAndNil(FHintMsgs);
  2256. FreeAndNil(FHintMsgsGood);
  2257. inherited Destroy;
  2258. end;
  2259. { TTestModule }
  2260. procedure TTestModule.TestReservedWords;
  2261. var
  2262. i: integer;
  2263. begin
  2264. for i:=low(JSReservedWords) to High(JSReservedWords)-1 do
  2265. if CompareStr(JSReservedWords[i],JSReservedWords[i+1])>=0 then
  2266. Fail('20170203135442 '+JSReservedWords[i]+' >= '+JSReservedWords[i+1]);
  2267. for i:=low(JSReservedGlobalWords) to High(JSReservedGlobalWords)-1 do
  2268. if CompareStr(JSReservedGlobalWords[i],JSReservedGlobalWords[i+1])>=0 then
  2269. Fail('20170203135443 '+JSReservedGlobalWords[i]+' >= '+JSReservedGlobalWords[i+1]);
  2270. end;
  2271. procedure TTestModule.TestEmptyProgram;
  2272. begin
  2273. StartProgram(false);
  2274. Add('begin');
  2275. ConvertProgram;
  2276. CheckSource('TestEmptyProgram','','');
  2277. end;
  2278. procedure TTestModule.TestEmptyProgramUseStrict;
  2279. begin
  2280. Converter.Options:=Converter.Options+[coUseStrict];
  2281. StartProgram(false);
  2282. Add('begin');
  2283. ConvertProgram;
  2284. CheckSource('TestEmptyProgramUseStrict','','');
  2285. end;
  2286. procedure TTestModule.TestEmptyUnit;
  2287. begin
  2288. StartUnit(false);
  2289. Add('interface');
  2290. Add('implementation');
  2291. ConvertUnit;
  2292. CheckSource('TestEmptyUnit',
  2293. LinesToStr([
  2294. ]),
  2295. '');
  2296. end;
  2297. procedure TTestModule.TestEmptyUnitUseStrict;
  2298. begin
  2299. Converter.Options:=Converter.Options+[coUseStrict];
  2300. StartUnit(false);
  2301. Add('interface');
  2302. Add('implementation');
  2303. ConvertUnit;
  2304. CheckSource('TestEmptyUnitUseStrict',
  2305. LinesToStr([
  2306. ''
  2307. ]),
  2308. '');
  2309. end;
  2310. procedure TTestModule.TestDottedUnitNames;
  2311. begin
  2312. AddModuleWithIntfImplSrc('NS1.Unit2.pas',
  2313. LinesToStr([
  2314. 'var iV: longint;'
  2315. ]),
  2316. '');
  2317. FFilename:='ns1.test1.pp';
  2318. StartProgram(true);
  2319. Add('uses unIt2;');
  2320. Add('var');
  2321. Add(' i: longint;');
  2322. Add('begin');
  2323. Add(' i:=iv;');
  2324. Add(' i:=uNit2.iv;');
  2325. Add(' i:=Ns1.TEst1.i;');
  2326. ConvertProgram;
  2327. CheckSource('TestDottedUnitNames',
  2328. LinesToStr([
  2329. 'this.i = 0;',
  2330. '']),
  2331. LinesToStr([ // this.$init
  2332. '$mod.i = pas["NS1.Unit2"].iV;',
  2333. '$mod.i = pas["NS1.Unit2"].iV;',
  2334. '$mod.i = $mod.i;',
  2335. '']) );
  2336. end;
  2337. procedure TTestModule.TestDottedUnitNameImpl;
  2338. begin
  2339. AddModuleWithIntfImplSrc('TEST.UnitA.pas',
  2340. LinesToStr([
  2341. 'type',
  2342. ' TObject = class end;',
  2343. ' TTestA = class',
  2344. ' end;'
  2345. ]),
  2346. LinesToStr(['uses TEST.UnitB;'])
  2347. );
  2348. AddModuleWithIntfImplSrc('TEST.UnitB.pas',
  2349. LinesToStr([
  2350. 'uses TEST.UnitA;',
  2351. 'type TTestB = class(TTestA);'
  2352. ]),
  2353. ''
  2354. );
  2355. StartProgram(true);
  2356. Add('uses TEST.UnitA;');
  2357. Add('begin');
  2358. ConvertProgram;
  2359. CheckSource('TestDottedUnitNameImpl',
  2360. LinesToStr([
  2361. '']),
  2362. LinesToStr([ // this.$init
  2363. '']) );
  2364. CheckUnit('TEST.UnitA.pas',
  2365. LinesToStr([
  2366. 'rtl.module("TEST.UnitA", ["system"], function () {',
  2367. ' var $mod = this;',
  2368. ' rtl.createClass(this, "TObject", null, function () {',
  2369. ' this.$init = function () {',
  2370. ' };',
  2371. ' this.$final = function () {',
  2372. ' };',
  2373. ' });',
  2374. ' rtl.createClass(this, "TTestA", this.TObject, function () {',
  2375. ' });',
  2376. '}, ["TEST.UnitB"]);'
  2377. ]));
  2378. CheckUnit('TEST.UnitB.pas',
  2379. LinesToStr([
  2380. 'rtl.module("TEST.UnitB", ["system","TEST.UnitA"], function () {',
  2381. ' var $mod = this;',
  2382. ' rtl.createClass(this, "TTestB", pas["TEST.UnitA"].TTestA, function () {',
  2383. ' });',
  2384. '});'
  2385. ]));
  2386. end;
  2387. procedure TTestModule.TestDottedUnitExpr;
  2388. begin
  2389. AddModuleWithIntfImplSrc('NS2.SubNs2.Unit2.pas',
  2390. LinesToStr([
  2391. 'procedure DoIt;'
  2392. ]),
  2393. 'procedure DoIt; begin end;');
  2394. FFilename:='Ns1.SubNs1.Test1.pp';
  2395. StartProgram(true);
  2396. Add('uses Ns2.sUbnS2.unIt2;');
  2397. Add('var');
  2398. Add(' i: longint;');
  2399. Add('begin');
  2400. Add(' ns2.subns2.unit2.doit;');
  2401. Add(' i:=Ns1.SubNS1.TEst1.i;');
  2402. ConvertProgram;
  2403. CheckSource('TestDottedUnitExpr',
  2404. LinesToStr([
  2405. 'this.i = 0;',
  2406. '']),
  2407. LinesToStr([ // this.$init
  2408. 'pas["NS2.SubNs2.Unit2"].DoIt();',
  2409. '$mod.i = $mod.i;',
  2410. '']) );
  2411. end;
  2412. procedure TTestModule.Test_ModeFPCFail;
  2413. begin
  2414. StartProgram(false);
  2415. Add('{$mode FPC}');
  2416. Add('begin');
  2417. SetExpectedScannerError('Invalid mode: "FPC"',nErrInvalidMode);
  2418. ConvertProgram;
  2419. end;
  2420. procedure TTestModule.Test_ModeSwitchCBlocksFail;
  2421. begin
  2422. StartProgram(false);
  2423. Add('{$modeswitch cblocks-}');
  2424. Add('begin');
  2425. ConvertProgram;
  2426. CheckHint(mtWarning,nErrInvalidModeSwitch,'Warning: test1.pp(3,23) : Invalid mode switch: "cblocks"');
  2427. CheckResolverUnexpectedHints();
  2428. end;
  2429. procedure TTestModule.TestUnit_UseSystem;
  2430. begin
  2431. StartUnit(true);
  2432. Add([
  2433. 'interface',
  2434. 'var i: integer;',
  2435. 'implementation']);
  2436. ConvertUnit;
  2437. CheckSource('TestUnit_UseSystem',
  2438. LinesToStr([
  2439. 'this.i = 0;',
  2440. '']),
  2441. LinesToStr([
  2442. '']) );
  2443. end;
  2444. procedure TTestModule.TestUnit_Intf1Impl2Intf1;
  2445. begin
  2446. AddModuleWithIntfImplSrc('unit1.pp',
  2447. LinesToStr([
  2448. 'type number = longint;']),
  2449. LinesToStr([
  2450. 'uses test1;',
  2451. 'procedure DoIt;',
  2452. 'begin',
  2453. ' i:=3;',
  2454. 'end;']));
  2455. StartUnit(true);
  2456. Add([
  2457. 'interface',
  2458. 'uses unit1;',
  2459. 'var i: number;',
  2460. 'implementation']);
  2461. ConvertUnit;
  2462. CheckSource('TestUnit_Intf1Impl2Intf1',
  2463. LinesToStr([
  2464. 'this.i = 0;',
  2465. '']),
  2466. LinesToStr([
  2467. '']) );
  2468. end;
  2469. procedure TTestModule.TestIncludeVersion;
  2470. begin
  2471. StartProgram(false);
  2472. Add([
  2473. 'var',
  2474. ' s: string;',
  2475. ' i: word;',
  2476. 'begin',
  2477. ' s:={$I %line%};',
  2478. ' i:={$I %linenum%};',
  2479. ' s:={$I %currentroutine%};',
  2480. ' s:={$I %pas2jsversion%};',
  2481. ' s:={$I %pas2jstarget%};',
  2482. ' s:={$I %pas2jstargetos%};',
  2483. ' s:={$I %pas2jstargetcpu%};',
  2484. ' s:={$I %file%};',
  2485. '']);
  2486. ConvertProgram;
  2487. CheckSource('TestIncludeVersion',
  2488. LinesToStr([
  2489. 'this.s="";',
  2490. 'this.i = 0;']),
  2491. LinesToStr([
  2492. '$mod.s = "7";',
  2493. '$mod.i = 8;',
  2494. '$mod.s = "<anonymous>";',
  2495. '$mod.s = "Comp.Ver.tcmodules";',
  2496. '$mod.s = "Browser";',
  2497. '$mod.s = "Browser";',
  2498. '$mod.s = "ECMAScript5";',
  2499. '$mod.s = "test1.pp";',
  2500. '']));
  2501. end;
  2502. procedure TTestModule.TestVarInt;
  2503. begin
  2504. StartProgram(false);
  2505. Add('var MyI: longint;');
  2506. Add('begin');
  2507. ConvertProgram;
  2508. CheckSource('TestVarInt','this.MyI=0;','');
  2509. end;
  2510. procedure TTestModule.TestVarBaseTypes;
  2511. begin
  2512. StartProgram(false);
  2513. Add('var');
  2514. Add(' i: longint;');
  2515. Add(' s: string;');
  2516. Add(' c: char;');
  2517. Add(' b: boolean;');
  2518. Add(' d: double;');
  2519. Add(' i2: longint = 3;');
  2520. Add(' s2: string = ''foo'';');
  2521. Add(' c2: char = ''4'';');
  2522. Add(' b2: boolean = true;');
  2523. Add(' d2: double = 5.6;');
  2524. Add(' i3: longint = $707;');
  2525. Add(' i4: nativeint = 9007199254740991;');
  2526. Add(' i5: nativeint = -9007199254740991-1;');
  2527. Add(' i6: nativeint = $fffffffffffff;');
  2528. Add(' i7: nativeint = -$fffffffffffff-1;');
  2529. Add(' i8: byte = 00;');
  2530. Add(' u8: nativeuint = $fffffffffffff;');
  2531. Add(' u9: nativeuint = $0000000000000;');
  2532. Add(' u10: nativeuint = $00ff00;');
  2533. Add('begin');
  2534. ConvertProgram;
  2535. CheckSource('TestVarBaseTypes',
  2536. LinesToStr([
  2537. 'this.i = 0;',
  2538. 'this.s = "";',
  2539. 'this.c = "";',
  2540. 'this.b = false;',
  2541. 'this.d = 0.0;',
  2542. 'this.i2 = 3;',
  2543. 'this.s2 = "foo";',
  2544. 'this.c2 = "4";',
  2545. 'this.b2 = true;',
  2546. 'this.d2 = 5.6;',
  2547. 'this.i3 = 0x707;',
  2548. 'this.i4 = 9007199254740991;',
  2549. 'this.i5 = -9007199254740991-1;',
  2550. 'this.i6 = 0xfffffffffffff;',
  2551. 'this.i7 =-0xfffffffffffff-1;',
  2552. 'this.i8 = 0;',
  2553. 'this.u8 = 0xfffffffffffff;',
  2554. 'this.u9 = 0x0;',
  2555. 'this.u10 = 0xff00;'
  2556. ]),
  2557. '');
  2558. end;
  2559. procedure TTestModule.TestBaseTypeSingleFail;
  2560. begin
  2561. StartProgram(false);
  2562. Add('var s: single;');
  2563. SetExpectedPasResolverError('identifier not found "single"',PasResolveEval.nIdentifierNotFound);
  2564. ConvertProgram;
  2565. end;
  2566. procedure TTestModule.TestBaseTypeExtendedFail;
  2567. begin
  2568. StartProgram(false);
  2569. Add('var e: extended;');
  2570. SetExpectedPasResolverError('identifier not found "extended"',PasResolveEval.nIdentifierNotFound);
  2571. ConvertProgram;
  2572. end;
  2573. procedure TTestModule.TestConstBaseTypes;
  2574. begin
  2575. StartProgram(false);
  2576. Add('const');
  2577. Add(' i: longint = 3;');
  2578. Add(' s: string = ''foo'';');
  2579. Add(' c: char = ''4'';');
  2580. Add(' b: boolean = true;');
  2581. Add(' d: double = 5.6;');
  2582. Add(' e = low(word);');
  2583. Add(' f = high(word);');
  2584. Add('begin');
  2585. ConvertProgram;
  2586. CheckSource('TestVarBaseTypes',
  2587. LinesToStr([
  2588. 'this.i=3;',
  2589. 'this.s="foo";',
  2590. 'this.c="4";',
  2591. 'this.b=true;',
  2592. 'this.d=5.6;',
  2593. 'this.e = 0;',
  2594. 'this.f = 65535;'
  2595. ]),
  2596. '');
  2597. end;
  2598. procedure TTestModule.TestAliasTypeRef;
  2599. begin
  2600. StartProgram(false);
  2601. Add('type');
  2602. Add(' a=longint;');
  2603. Add(' b=a;');
  2604. Add('var');
  2605. Add(' c: A;');
  2606. Add(' d: B;');
  2607. Add('begin');
  2608. ConvertProgram;
  2609. CheckSource('TestAliasTypeRef',
  2610. LinesToStr([ // statements
  2611. 'this.c = 0;',
  2612. 'this.d = 0;'
  2613. ]),
  2614. LinesToStr([ // this.$main
  2615. ''
  2616. ]));
  2617. end;
  2618. procedure TTestModule.TestTypeCast_BaseTypes;
  2619. begin
  2620. StartProgram(false);
  2621. Add([
  2622. 'var',
  2623. ' i: longint;',
  2624. ' b: boolean;',
  2625. ' d: double;',
  2626. ' s: string;',
  2627. ' c: char;',
  2628. 'begin',
  2629. ' i:=longint(i);',
  2630. ' i:=longint(b);',
  2631. ' b:=boolean(b);',
  2632. ' b:=boolean(i);',
  2633. ' d:=double(d);',
  2634. ' d:=double(i);',
  2635. ' s:=string(s);',
  2636. ' s:=string(c);',
  2637. ' c:=char(c);',
  2638. ' c:=char(i);',
  2639. ' c:=char(65);',
  2640. ' c:=char(#10);',
  2641. ' c:=char(#$E000);',
  2642. '']);
  2643. ConvertProgram;
  2644. CheckSource('TestAliasTypeRef',
  2645. LinesToStr([ // statements
  2646. 'this.i = 0;',
  2647. 'this.b = false;',
  2648. 'this.d = 0.0;',
  2649. 'this.s = "";',
  2650. 'this.c = "";',
  2651. '']),
  2652. LinesToStr([ // this.$main
  2653. '$mod.i = $mod.i;',
  2654. '$mod.i = ($mod.b ? 1 : 0);',
  2655. '$mod.b = $mod.b;',
  2656. '$mod.b = $mod.i != 0;',
  2657. '$mod.d = $mod.d;',
  2658. '$mod.d = $mod.i;',
  2659. '$mod.s = $mod.s;',
  2660. '$mod.s = $mod.c;',
  2661. '$mod.c = $mod.c;',
  2662. '$mod.c = String.fromCharCode($mod.i);',
  2663. '$mod.c = "A";',
  2664. '$mod.c = "\n";',
  2665. '$mod.c = "";',
  2666. '']));
  2667. end;
  2668. procedure TTestModule.TestTypeCast_AliasBaseTypes;
  2669. begin
  2670. StartProgram(false);
  2671. Add('type');
  2672. Add(' integer = longint;');
  2673. Add(' TYesNo = boolean;');
  2674. Add(' TFloat = double;');
  2675. Add(' TCaption = string;');
  2676. Add(' TChar = char;');
  2677. Add('var');
  2678. Add(' i: integer;');
  2679. Add(' b: TYesNo;');
  2680. Add(' d: TFloat;');
  2681. Add(' s: TCaption;');
  2682. Add(' c: TChar;');
  2683. Add('begin');
  2684. Add(' i:=integer(i);');
  2685. Add(' i:=integer(b);');
  2686. Add(' b:=TYesNo(b);');
  2687. Add(' b:=TYesNo(i);');
  2688. Add(' d:=TFloat(d);');
  2689. Add(' d:=TFloat(i);');
  2690. Add(' s:=TCaption(s);');
  2691. Add(' s:=TCaption(c);');
  2692. Add(' c:=TChar(c);');
  2693. ConvertProgram;
  2694. CheckSource('TestAliasTypeRef',
  2695. LinesToStr([ // statements
  2696. 'this.i = 0;',
  2697. 'this.b = false;',
  2698. 'this.d = 0.0;',
  2699. 'this.s = "";',
  2700. 'this.c = "";',
  2701. '']),
  2702. LinesToStr([ // this.$main
  2703. '$mod.i = $mod.i;',
  2704. '$mod.i = ($mod.b ? 1 : 0);',
  2705. '$mod.b = $mod.b;',
  2706. '$mod.b = $mod.i != 0;',
  2707. '$mod.d = $mod.d;',
  2708. '$mod.d = $mod.i;',
  2709. '$mod.s = $mod.s;',
  2710. '$mod.s = $mod.c;',
  2711. '$mod.c = $mod.c;',
  2712. '']));
  2713. end;
  2714. procedure TTestModule.TestEmptyProc;
  2715. begin
  2716. StartProgram(false);
  2717. Add('procedure Test;');
  2718. Add('begin');
  2719. Add('end;');
  2720. Add('begin');
  2721. ConvertProgram;
  2722. CheckSource('TestEmptyProc',
  2723. LinesToStr([ // statements
  2724. 'this.Test = function () {',
  2725. '};'
  2726. ]),
  2727. LinesToStr([ // this.$main
  2728. ''
  2729. ]));
  2730. end;
  2731. procedure TTestModule.TestProcOneParam;
  2732. begin
  2733. StartProgram(false);
  2734. Add('procedure ProcA(i: longint);');
  2735. Add('begin');
  2736. Add('end;');
  2737. Add('begin');
  2738. Add(' PROCA(3);');
  2739. ConvertProgram;
  2740. CheckSource('TestProcOneParam',
  2741. LinesToStr([ // statements
  2742. 'this.ProcA = function (i) {',
  2743. '};'
  2744. ]),
  2745. LinesToStr([ // this.$main
  2746. '$mod.ProcA(3);'
  2747. ]));
  2748. end;
  2749. procedure TTestModule.TestFunctionWithoutParams;
  2750. begin
  2751. StartProgram(false);
  2752. Add('function FuncA: longint;');
  2753. Add('begin');
  2754. Add('end;');
  2755. Add('var i: longint;');
  2756. Add('begin');
  2757. Add(' I:=FUNCA();');
  2758. Add(' I:=FUNCA;');
  2759. Add(' FUNCA();');
  2760. Add(' FUNCA;');
  2761. ConvertProgram;
  2762. CheckSource('TestProcWithoutParams',
  2763. LinesToStr([ // statements
  2764. 'this.FuncA = function () {',
  2765. ' var Result = 0;',
  2766. ' return Result;',
  2767. '};',
  2768. 'this.i=0;'
  2769. ]),
  2770. LinesToStr([ // this.$main
  2771. '$mod.i=$mod.FuncA();',
  2772. '$mod.i=$mod.FuncA();',
  2773. '$mod.FuncA();',
  2774. '$mod.FuncA();'
  2775. ]));
  2776. end;
  2777. procedure TTestModule.TestProcedureWithoutParams;
  2778. begin
  2779. StartProgram(false);
  2780. Add('procedure ProcA;');
  2781. Add('begin');
  2782. Add('end;');
  2783. Add('begin');
  2784. Add(' PROCA();');
  2785. Add(' PROCA;');
  2786. ConvertProgram;
  2787. CheckSource('TestProcWithoutParams',
  2788. LinesToStr([ // statements
  2789. 'this.ProcA = function () {',
  2790. '};'
  2791. ]),
  2792. LinesToStr([ // this.$main
  2793. '$mod.ProcA();',
  2794. '$mod.ProcA();'
  2795. ]));
  2796. end;
  2797. procedure TTestModule.TestIncDec;
  2798. begin
  2799. StartProgram(false);
  2800. Add([
  2801. 'procedure DoIt(var i: longint);',
  2802. 'begin',
  2803. ' inc(i);',
  2804. ' inc(i,2);',
  2805. 'end;',
  2806. 'var',
  2807. ' Bar: longint;',
  2808. 'begin',
  2809. ' inc(bar);',
  2810. ' inc(bar,2);',
  2811. ' dec(bar);',
  2812. ' dec(bar,3);',
  2813. '']);
  2814. ConvertProgram;
  2815. CheckSource('TestIncDec',
  2816. LinesToStr([ // statements
  2817. 'this.DoIt = function (i) {',
  2818. ' i.set(i.get()+1);',
  2819. ' i.set(i.get()+2);',
  2820. '};',
  2821. 'this.Bar = 0;'
  2822. ]),
  2823. LinesToStr([ // this.$main
  2824. '$mod.Bar+=1;',
  2825. '$mod.Bar+=2;',
  2826. '$mod.Bar-=1;',
  2827. '$mod.Bar-=3;'
  2828. ]));
  2829. end;
  2830. procedure TTestModule.TestLoHiFpcMode;
  2831. begin
  2832. StartProgram(false);
  2833. Add([
  2834. '{$mode objfpc}',
  2835. 'const',
  2836. ' LoByte1 = Lo(Word($1234));',
  2837. ' HiByte1 = Hi(Word($1234));',
  2838. ' LoByte2 = Lo(SmallInt($1234));',
  2839. ' HiByte2 = Hi(SmallInt($1234));',
  2840. ' LoWord1 = Lo($1234CDEF);',
  2841. ' HiWord1 = Hi($1234CDEF);',
  2842. ' LoWord2 = Lo(-$1234CDEF);',
  2843. ' HiWord2 = Hi(-$1234CDEF);',
  2844. ' lo4:byte=lo(byte($34));',
  2845. ' hi4:byte=hi(byte($34));',
  2846. ' lo5:byte=lo(shortint(-$34));',
  2847. ' hi5:byte=hi(shortint(-$34));',
  2848. ' lo6:longword=lo($123456789ABCD);',
  2849. ' hi6:longword=hi($123456789ABCD);',
  2850. ' lo7:longword=lo(-$123456789ABCD);',
  2851. ' hi7:longword=hi(-$123456789ABCD);',
  2852. 'var',
  2853. ' b: Byte;',
  2854. ' ss: shortint;',
  2855. ' w: Word;',
  2856. ' si: SmallInt;',
  2857. ' lw: LongWord;',
  2858. ' li: LongInt;',
  2859. ' b2: Byte;',
  2860. ' ni: nativeint;',
  2861. 'begin',
  2862. ' w := $1234;',
  2863. ' ss := -$12;',
  2864. ' b := lo(ss);',
  2865. ' b := HI(ss);',
  2866. ' b := lo(w);',
  2867. ' b := HI(w);',
  2868. ' b2 := lo(b);',
  2869. ' b2 := hi(b);',
  2870. ' lw := $1234CDEF;',
  2871. ' w := lo(lw);',
  2872. ' w := hi(lw);',
  2873. ' ni := $123456789ABCD;',
  2874. ' lw := lo(ni);',
  2875. ' lw := hi(ni);',
  2876. '']);
  2877. ConvertProgram;
  2878. CheckSource('TestLoHiFpcMode',
  2879. LinesToStr([ // statements
  2880. 'this.LoByte1 = 0x1234 & 0xFF;',
  2881. 'this.HiByte1 = (0x1234 >> 8) & 0xFF;',
  2882. 'this.LoByte2 = 0x1234 & 0xFF;',
  2883. 'this.HiByte2 = (0x1234 >> 8) & 0xFF;',
  2884. 'this.LoWord1 = 0x1234CDEF & 0xFFFF;',
  2885. 'this.HiWord1 = (0x1234CDEF >> 16) & 0xFFFF;',
  2886. 'this.LoWord2 = -0x1234CDEF & 0xFFFF;',
  2887. 'this.HiWord2 = (-0x1234CDEF >> 16) & 0xFFFF;',
  2888. 'this.lo4 = 0x34 & 0xF;',
  2889. 'this.hi4 = (0x34 >> 4) & 0xF;',
  2890. 'this.lo5 = (((-0x34 & 255) << 24) >> 24) & 0xFF;',
  2891. 'this.hi5 = ((((-0x34 & 255) << 24) >> 24) >> 8) & 0xFF;',
  2892. 'this.lo6 = 0x123456789ABCD >>> 0;',
  2893. 'this.hi6 = 74565 >>> 0;',
  2894. 'this.lo7 = -0x123456789ABCD >>> 0;',
  2895. 'this.hi7 = Math.floor(-0x123456789ABCD / 4294967296) >>> 0;',
  2896. 'this.b = 0;',
  2897. 'this.ss = 0;',
  2898. 'this.w = 0;',
  2899. 'this.si = 0;',
  2900. 'this.lw = 0;',
  2901. 'this.li = 0;',
  2902. 'this.b2 = 0;',
  2903. 'this.ni = 0;',
  2904. '']),
  2905. LinesToStr([ // this.$main
  2906. '$mod.w = 0x1234;',
  2907. '$mod.ss = -0x12;',
  2908. '$mod.b = $mod.ss & 0xFF;',
  2909. '$mod.b = ($mod.ss >> 8) & 0xFF;',
  2910. '$mod.b = $mod.w & 0xFF;',
  2911. '$mod.b = ($mod.w >> 8) & 0xFF;',
  2912. '$mod.b2 = $mod.b & 0xF;',
  2913. '$mod.b2 = ($mod.b >> 4) & 0xF;',
  2914. '$mod.lw = 0x1234CDEF;',
  2915. '$mod.w = $mod.lw & 0xFFFF;',
  2916. '$mod.w = ($mod.lw >> 16) & 0xFFFF;',
  2917. '$mod.ni = 0x123456789ABCD;',
  2918. '$mod.lw = $mod.ni >>> 0;',
  2919. '$mod.lw = Math.floor($mod.ni / 4294967296) >>> 0;',
  2920. '']));
  2921. end;
  2922. procedure TTestModule.TestLoHiDelphiMode;
  2923. begin
  2924. StartProgram(false);
  2925. Add([
  2926. '{$mode delphi}',
  2927. 'const',
  2928. ' LoByte1 = Lo(Word($1234));',
  2929. ' HiByte1 = Hi(Word($1234));',
  2930. ' LoByte2 = Lo(SmallInt($1234));',
  2931. ' HiByte2 = Hi(SmallInt($1234));',
  2932. ' LoByte3 = Lo($1234CDEF);',
  2933. ' HiByte3 = Hi($1234CDEF);',
  2934. ' LoByte4 = Lo(-$1234CDEF);',
  2935. ' HiByte4 = Hi(-$1234CDEF);',
  2936. 'var',
  2937. ' b: Byte;',
  2938. ' w: Word;',
  2939. ' si: SmallInt;',
  2940. ' lw: LongWord;',
  2941. ' li: LongInt;',
  2942. 'begin',
  2943. ' w := $1234;',
  2944. ' b := lo(w);',
  2945. ' b := HI(w);',
  2946. ' lw := $1234CDEF;',
  2947. ' b := lo(lw);',
  2948. ' b := hi(lw);',
  2949. '']);
  2950. ConvertProgram;
  2951. CheckSource('TestLoHiDelphiMode',
  2952. LinesToStr([ // statements
  2953. 'this.LoByte1 = 0x1234 & 0xFF;',
  2954. 'this.HiByte1 = (0x1234 >> 8) & 0xFF;',
  2955. 'this.LoByte2 = 0x1234 & 0xFF;',
  2956. 'this.HiByte2 = (0x1234 >> 8) & 0xFF;',
  2957. 'this.LoByte3 = 0x1234CDEF & 0xFF;',
  2958. 'this.HiByte3 = (0x1234CDEF >> 8) & 0xFF;',
  2959. 'this.LoByte4 = -0x1234CDEF & 0xFF;',
  2960. 'this.HiByte4 = (-0x1234CDEF >> 8) & 0xFF;',
  2961. 'this.b = 0;',
  2962. 'this.w = 0;',
  2963. 'this.si = 0;',
  2964. 'this.lw = 0;',
  2965. 'this.li = 0;'
  2966. ]),
  2967. LinesToStr([ // this.$main
  2968. '$mod.w = 0x1234;',
  2969. '$mod.b = $mod.w & 0xFF;',
  2970. '$mod.b = ($mod.w >> 8) & 0xFF;',
  2971. '$mod.lw = 0x1234CDEF;',
  2972. '$mod.b = $mod.lw & 0xFF;',
  2973. '$mod.b = ($mod.lw >> 8) & 0xFF;'
  2974. ]));
  2975. end;
  2976. procedure TTestModule.TestAssignments;
  2977. begin
  2978. StartProgram(false);
  2979. Parser.Options:=Parser.Options+[po_cassignments];
  2980. Add('var');
  2981. Add(' Bar:longint;');
  2982. Add('begin');
  2983. Add(' bar:=3;');
  2984. Add(' bar+=4;');
  2985. Add(' bar-=5;');
  2986. Add(' bar*=6;');
  2987. ConvertProgram;
  2988. CheckSource('TestAssignments',
  2989. LinesToStr([ // statements
  2990. 'this.Bar = 0;'
  2991. ]),
  2992. LinesToStr([ // this.$main
  2993. '$mod.Bar=3;',
  2994. '$mod.Bar+=4;',
  2995. '$mod.Bar-=5;',
  2996. '$mod.Bar*=6;'
  2997. ]));
  2998. end;
  2999. procedure TTestModule.TestArithmeticOperators1;
  3000. begin
  3001. StartProgram(false);
  3002. Add('var');
  3003. Add(' vA,vB,vC:longint;');
  3004. Add('begin');
  3005. Add(' va:=1;');
  3006. Add(' vb:=va+va;');
  3007. Add(' vb:=va div vb;');
  3008. Add(' vb:=va mod vb;');
  3009. Add(' vb:=va+va*vb+va div vb;');
  3010. Add(' vc:=-va;');
  3011. Add(' va:=va-vb;');
  3012. Add(' vb:=va;');
  3013. Add(' if va<vb then vc:=va else vc:=vb;');
  3014. ConvertProgram;
  3015. CheckSource('TestArithmeticOperators1',
  3016. LinesToStr([ // statements
  3017. 'this.vA = 0;',
  3018. 'this.vB = 0;',
  3019. 'this.vC = 0;'
  3020. ]),
  3021. LinesToStr([ // this.$main
  3022. '$mod.vA = 1;',
  3023. '$mod.vB = $mod.vA + $mod.vA;',
  3024. '$mod.vB = rtl.trunc($mod.vA / $mod.vB);',
  3025. '$mod.vB = $mod.vA % $mod.vB;',
  3026. '$mod.vB = $mod.vA + ($mod.vA * $mod.vB) + rtl.trunc($mod.vA / $mod.vB);',
  3027. '$mod.vC = -$mod.vA;',
  3028. '$mod.vA = $mod.vA - $mod.vB;',
  3029. '$mod.vB = $mod.vA;',
  3030. 'if ($mod.vA < $mod.vB){ $mod.vC = $mod.vA } else $mod.vC = $mod.vB;'
  3031. ]));
  3032. end;
  3033. procedure TTestModule.TestLogicalOperators;
  3034. begin
  3035. StartProgram(false);
  3036. Add('var');
  3037. Add(' vA,vB,vC:boolean;');
  3038. Add('begin');
  3039. Add(' va:=vb and vc;');
  3040. Add(' va:=vb or vc;');
  3041. Add(' va:=vb xor vc;');
  3042. Add(' va:=true and vc;');
  3043. Add(' va:=(vb and vc) or (va and vb);');
  3044. Add(' va:=not vb;');
  3045. ConvertProgram;
  3046. CheckSource('TestLogicalOperators',
  3047. LinesToStr([ // statements
  3048. 'this.vA = false;',
  3049. 'this.vB = false;',
  3050. 'this.vC = false;'
  3051. ]),
  3052. LinesToStr([ // this.$main
  3053. '$mod.vA = $mod.vB && $mod.vC;',
  3054. '$mod.vA = $mod.vB || $mod.vC;',
  3055. '$mod.vA = $mod.vB ^ $mod.vC;',
  3056. '$mod.vA = true && $mod.vC;',
  3057. '$mod.vA = ($mod.vB && $mod.vC) || ($mod.vA && $mod.vB);',
  3058. '$mod.vA = !$mod.vB;'
  3059. ]));
  3060. end;
  3061. procedure TTestModule.TestBitwiseOperators;
  3062. begin
  3063. StartProgram(false);
  3064. Add([
  3065. 'var',
  3066. ' vA,vB,vC:longint;',
  3067. ' X,Y,Z: nativeint;',
  3068. 'begin',
  3069. ' va:=vb and vc;',
  3070. ' va:=vb or vc;',
  3071. ' va:=vb xor vc;',
  3072. ' va:=vb shl vc;',
  3073. ' va:=vb shr vc;',
  3074. ' va:=3 and vc;',
  3075. ' va:=(vb and vc) or (va and vb);',
  3076. ' va:=not vb;',
  3077. ' X:=Y and Z;',
  3078. ' X:=Y and va;',
  3079. ' X:=Y or Z;',
  3080. ' X:=Y or va;',
  3081. ' X:=Y xor Z;',
  3082. ' X:=Y xor va;',
  3083. '']);
  3084. ConvertProgram;
  3085. CheckSource('TestBitwiseOperators',
  3086. LinesToStr([ // statements
  3087. 'this.vA = 0;',
  3088. 'this.vB = 0;',
  3089. 'this.vC = 0;',
  3090. 'this.X = 0;',
  3091. 'this.Y = 0;',
  3092. 'this.Z = 0;',
  3093. '']),
  3094. LinesToStr([ // this.$main
  3095. '$mod.vA = $mod.vB & $mod.vC;',
  3096. '$mod.vA = $mod.vB | $mod.vC;',
  3097. '$mod.vA = $mod.vB ^ $mod.vC;',
  3098. '$mod.vA = $mod.vB << $mod.vC;',
  3099. '$mod.vA = $mod.vB >>> $mod.vC;',
  3100. '$mod.vA = 3 & $mod.vC;',
  3101. '$mod.vA = ($mod.vB & $mod.vC) | ($mod.vA & $mod.vB);',
  3102. '$mod.vA = ~$mod.vB;',
  3103. '$mod.X = rtl.and($mod.Y, $mod.Z);',
  3104. '$mod.X = $mod.Y & $mod.vA;',
  3105. '$mod.X = rtl.or($mod.Y, $mod.Z);',
  3106. '$mod.X = rtl.or($mod.Y, $mod.vA);',
  3107. '$mod.X = rtl.xor($mod.Y, $mod.Z);',
  3108. '$mod.X = rtl.xor($mod.Y, $mod.vA);',
  3109. '']));
  3110. end;
  3111. procedure TTestModule.TestBitwiseOperatorsLongword;
  3112. begin
  3113. StartProgram(false);
  3114. Add([
  3115. 'var',
  3116. ' a,b,c:longword;',
  3117. ' i: longint;',
  3118. 'begin',
  3119. ' a:=$12345678;',
  3120. ' b:=$EDCBA987;',
  3121. ' c:=not a;',
  3122. ' c:=a and b;',
  3123. ' c:=a and $ffff0000;',
  3124. ' c:=a or b;',
  3125. ' c:=a or $ff00ff00;',
  3126. ' c:=a xor b;',
  3127. ' c:=a xor $f0f0f0f0;',
  3128. ' c:=a shl 1;',
  3129. ' c:=a shl 16;',
  3130. ' c:=a shl 24;',
  3131. ' c:=a shl b;',
  3132. ' c:=a shr 1;',
  3133. ' c:=a shr 16;',
  3134. ' c:=a shr 24;',
  3135. ' c:=a shr b;',
  3136. ' c:=(b and c) or (a and b);',
  3137. ' c:=i and a;',
  3138. ' c:=i or a;',
  3139. ' c:=i xor a;',
  3140. '']);
  3141. ConvertProgram;
  3142. CheckSource('TestBitwiseOperatorsLongword',
  3143. LinesToStr([ // statements
  3144. 'this.a = 0;',
  3145. 'this.b = 0;',
  3146. 'this.c = 0;',
  3147. 'this.i = 0;',
  3148. '']),
  3149. LinesToStr([ // this.$main
  3150. '$mod.a = 0x12345678;',
  3151. '$mod.b = 0xEDCBA987;',
  3152. '$mod.c = rtl.lw(~$mod.a);',
  3153. '$mod.c = rtl.lw($mod.a & $mod.b);',
  3154. '$mod.c = rtl.lw($mod.a & 0xffff0000);',
  3155. '$mod.c = rtl.lw($mod.a | $mod.b);',
  3156. '$mod.c = rtl.lw($mod.a | 0xff00ff00);',
  3157. '$mod.c = rtl.lw($mod.a ^ $mod.b);',
  3158. '$mod.c = rtl.lw($mod.a ^ 0xf0f0f0f0);',
  3159. '$mod.c = rtl.lw($mod.a << 1);',
  3160. '$mod.c = rtl.lw($mod.a << 16);',
  3161. '$mod.c = rtl.lw($mod.a << 24);',
  3162. '$mod.c = rtl.lw($mod.a << $mod.b);',
  3163. '$mod.c = rtl.lw($mod.a >>> 1);',
  3164. '$mod.c = rtl.lw($mod.a >>> 16);',
  3165. '$mod.c = rtl.lw($mod.a >>> 24);',
  3166. '$mod.c = rtl.lw($mod.a >>> $mod.b);',
  3167. '$mod.c = rtl.lw(rtl.lw($mod.b & $mod.c) | rtl.lw($mod.a & $mod.b));',
  3168. '$mod.c = $mod.i & $mod.a;',
  3169. '$mod.c = $mod.i | $mod.a;',
  3170. '$mod.c = $mod.i ^ $mod.a;',
  3171. '']));
  3172. end;
  3173. procedure TTestModule.TestPrgProcVar;
  3174. begin
  3175. StartProgram(false);
  3176. Add('procedure Proc1;');
  3177. Add('type');
  3178. Add(' t1=longint;');
  3179. Add('var');
  3180. Add(' vA:t1;');
  3181. Add('begin');
  3182. Add('end;');
  3183. Add('begin');
  3184. ConvertProgram;
  3185. CheckSource('TestPrgProcVar',
  3186. LinesToStr([ // statements
  3187. 'this.Proc1 = function () {',
  3188. ' var vA=0;',
  3189. '};'
  3190. ]),
  3191. LinesToStr([ // this.$main
  3192. ''
  3193. ]));
  3194. end;
  3195. procedure TTestModule.TestUnitProcVar;
  3196. begin
  3197. StartUnit(false);
  3198. Add('interface');
  3199. Add('');
  3200. Add('type tA=string; // unit scope');
  3201. Add('procedure Proc1;');
  3202. Add('');
  3203. Add('implementation');
  3204. Add('');
  3205. Add('procedure Proc1;');
  3206. Add('type tA=longint; // local proc scope');
  3207. Add('var v1:tA; // using local tA');
  3208. Add('begin');
  3209. Add('end;');
  3210. Add('var v2:tA; // using interface tA');
  3211. ConvertUnit;
  3212. CheckSource('TestUnitProcVar',
  3213. LinesToStr([ // statements
  3214. 'var $impl = $mod.$impl;',
  3215. 'this.Proc1 = function () {',
  3216. ' var v1 = 0;',
  3217. '};',
  3218. '']),
  3219. // this.$init
  3220. '',
  3221. // implementation
  3222. LinesToStr([
  3223. '$impl.v2 = "";',
  3224. '']));
  3225. end;
  3226. procedure TTestModule.TestImplProc;
  3227. begin
  3228. StartUnit(false);
  3229. Add('interface');
  3230. Add('');
  3231. Add('procedure Proc1;');
  3232. Add('');
  3233. Add('implementation');
  3234. Add('');
  3235. Add('procedure Proc1; begin end;');
  3236. Add('procedure Proc2; begin end;');
  3237. Add('initialization');
  3238. Add(' Proc1;');
  3239. Add(' Proc2;');
  3240. ConvertUnit;
  3241. CheckSource('TestImplProc',
  3242. LinesToStr([ // statements
  3243. 'var $impl = $mod.$impl;',
  3244. 'this.Proc1 = function () {',
  3245. '};',
  3246. '']),
  3247. LinesToStr([ // this.$init
  3248. '$mod.Proc1();',
  3249. '$impl.Proc2();',
  3250. '']),
  3251. LinesToStr([ // implementation
  3252. '$impl.Proc2 = function () {',
  3253. '};',
  3254. ''])
  3255. );
  3256. end;
  3257. procedure TTestModule.TestFunctionResult;
  3258. begin
  3259. StartProgram(false);
  3260. Add('function Func1: longint;');
  3261. Add('begin');
  3262. Add(' Result:=3;');
  3263. Add(' Func1:=4;');
  3264. Add('end;');
  3265. Add('begin');
  3266. ConvertProgram;
  3267. CheckSource('TestFunctionResult',
  3268. LinesToStr([ // statements
  3269. 'this.Func1 = function () {',
  3270. ' var Result = 0;',
  3271. ' Result = 3;',
  3272. ' Result = 4;',
  3273. ' return Result;',
  3274. '};'
  3275. ]),
  3276. '');
  3277. end;
  3278. procedure TTestModule.TestNestedProc;
  3279. begin
  3280. StartProgram(false);
  3281. Add([
  3282. 'var vInUnit: longint;',
  3283. 'function DoIt(pA,pD: longint): longint;',
  3284. 'var',
  3285. ' vB: longint;',
  3286. ' vC: longint;',
  3287. ' function Nesty(pA: longint): longint; ',
  3288. ' var vB: longint;',
  3289. ' begin',
  3290. ' Result:=pa+vb+vc+pd+vInUnit;',
  3291. ' nesty:=3;',
  3292. ' doit:=4;',
  3293. ' exit;',
  3294. ' end;',
  3295. 'begin',
  3296. ' Result:=pa+vb+vc;',
  3297. ' doit:=6;',
  3298. ' exit;',
  3299. 'end;',
  3300. 'begin']);
  3301. ConvertProgram;
  3302. CheckSource('TestNestedProc',
  3303. LinesToStr([ // statements
  3304. 'this.vInUnit = 0;',
  3305. 'this.DoIt = function (pA, pD) {',
  3306. ' var Result = 0;',
  3307. ' var vB = 0;',
  3308. ' var vC = 0;',
  3309. ' function Nesty(pA) {',
  3310. ' var Result$1 = 0;',
  3311. ' var vB = 0;',
  3312. ' Result$1 = pA + vB + vC + pD + $mod.vInUnit;',
  3313. ' Result$1 = 3;',
  3314. ' Result = 4;',
  3315. ' return Result$1;',
  3316. ' return Result$1;',
  3317. ' };',
  3318. ' Result = pA + vB + vC;',
  3319. ' Result = 6;',
  3320. ' return Result;',
  3321. ' return Result;',
  3322. '};'
  3323. ]),
  3324. '');
  3325. end;
  3326. procedure TTestModule.TestNestedProc_ResultString;
  3327. begin
  3328. StartProgram(false);
  3329. Add([
  3330. 'function DoIt: string;',
  3331. ' function Nesty: string; ',
  3332. ' begin',
  3333. ' nesty:=#65#66;',
  3334. ' nesty[1]:=#67;',
  3335. ' doit:=#68;',
  3336. ' doit[2]:=#69;',
  3337. ' end;',
  3338. 'begin',
  3339. ' doit:=#70;',
  3340. ' doit[3]:=#71;',
  3341. 'end;',
  3342. 'begin']);
  3343. ConvertProgram;
  3344. CheckSource('TestNestedProc_ResultString',
  3345. LinesToStr([ // statements
  3346. 'this.DoIt = function () {',
  3347. ' var Result = "";',
  3348. ' function Nesty() {',
  3349. ' var Result$1 = "";',
  3350. ' Result$1 = "AB";',
  3351. ' Result$1 = rtl.setCharAt(Result$1, 0, "C");',
  3352. ' Result = "D";',
  3353. ' Result = rtl.setCharAt(Result, 1, "E");',
  3354. ' return Result$1;',
  3355. ' };',
  3356. ' Result = "F";',
  3357. ' Result = rtl.setCharAt(Result, 2, "G");',
  3358. ' return Result;',
  3359. '};'
  3360. ]),
  3361. '');
  3362. end;
  3363. procedure TTestModule.TestForwardProc;
  3364. begin
  3365. StartProgram(false);
  3366. Add('procedure FuncA(Bar: longint); forward;');
  3367. Add('procedure FuncB(Bar: longint);');
  3368. Add('begin');
  3369. Add(' funca(bar);');
  3370. Add('end;');
  3371. Add('procedure funca(bar: longint);');
  3372. Add('begin');
  3373. Add(' if bar=3 then ;');
  3374. Add('end;');
  3375. Add('begin');
  3376. Add(' funca(4);');
  3377. Add(' funcb(5);');
  3378. ConvertProgram;
  3379. CheckSource('TestForwardProc',
  3380. LinesToStr([ // statements'
  3381. 'this.FuncB = function (Bar) {',
  3382. ' $mod.FuncA(Bar);',
  3383. '};',
  3384. 'this.FuncA = function (Bar) {',
  3385. ' if (Bar === 3);',
  3386. '};'
  3387. ]),
  3388. LinesToStr([
  3389. '$mod.FuncA(4);',
  3390. '$mod.FuncB(5);'
  3391. ])
  3392. );
  3393. end;
  3394. procedure TTestModule.TestNestedForwardProc;
  3395. begin
  3396. StartProgram(false);
  3397. Add('procedure FuncA;');
  3398. Add(' procedure FuncB(i: longint); forward;');
  3399. Add(' procedure FuncC(i: longint);');
  3400. Add(' begin');
  3401. Add(' funcb(i);');
  3402. Add(' end;');
  3403. Add(' procedure FuncB(i: longint);');
  3404. Add(' begin');
  3405. Add(' if i=3 then ;');
  3406. Add(' end;');
  3407. Add('begin');
  3408. Add(' funcc(4)');
  3409. Add('end;');
  3410. Add('begin');
  3411. Add(' funca;');
  3412. ConvertProgram;
  3413. CheckSource('TestNestedForwardProc',
  3414. LinesToStr([ // statements'
  3415. 'this.FuncA = function () {',
  3416. ' function FuncC(i) {',
  3417. ' FuncB(i);',
  3418. ' };',
  3419. ' function FuncB(i) {',
  3420. ' if (i === 3);',
  3421. ' };',
  3422. ' FuncC(4);',
  3423. '};'
  3424. ]),
  3425. LinesToStr([
  3426. '$mod.FuncA();'
  3427. ])
  3428. );
  3429. end;
  3430. procedure TTestModule.TestAssignFunctionResult;
  3431. begin
  3432. StartProgram(false);
  3433. Add('function Func1: longint;');
  3434. Add('begin');
  3435. Add('end;');
  3436. Add('var i: longint;');
  3437. Add('begin');
  3438. Add(' i:=func1();');
  3439. Add(' i:=func1()+func1();');
  3440. ConvertProgram;
  3441. CheckSource('TestAssignFunctionResult',
  3442. LinesToStr([ // statements
  3443. 'this.Func1 = function () {',
  3444. ' var Result = 0;',
  3445. ' return Result;',
  3446. '};',
  3447. 'this.i = 0;'
  3448. ]),
  3449. LinesToStr([
  3450. '$mod.i = $mod.Func1();',
  3451. '$mod.i = $mod.Func1() + $mod.Func1();'
  3452. ]));
  3453. end;
  3454. procedure TTestModule.TestFunctionResultInCondition;
  3455. begin
  3456. StartProgram(false);
  3457. Add('function Func1: longint;');
  3458. Add('begin');
  3459. Add('end;');
  3460. Add('function Func2: boolean;');
  3461. Add('begin');
  3462. Add('end;');
  3463. Add('var i: longint;');
  3464. Add('begin');
  3465. Add(' if func2 then ;');
  3466. Add(' if i=func1() then ;');
  3467. Add(' if i=func1 then ;');
  3468. ConvertProgram;
  3469. CheckSource('TestFunctionResultInCondition',
  3470. LinesToStr([ // statements
  3471. 'this.Func1 = function () {',
  3472. ' var Result = 0;',
  3473. ' return Result;',
  3474. '};',
  3475. 'this.Func2 = function () {',
  3476. ' var Result = false;',
  3477. ' return Result;',
  3478. '};',
  3479. 'this.i = 0;'
  3480. ]),
  3481. LinesToStr([
  3482. 'if ($mod.Func2());',
  3483. 'if ($mod.i === $mod.Func1());',
  3484. 'if ($mod.i === $mod.Func1());'
  3485. ]));
  3486. end;
  3487. procedure TTestModule.TestFunctionResultInForLoop;
  3488. begin
  3489. StartProgram(false);
  3490. Add([
  3491. 'function Func1(a: array of longint): longint;',
  3492. 'begin',
  3493. ' for Result:=High(a) downto Low(a) do if a[Result]=0 then exit;',
  3494. ' for Result in a do if a[Result]=0 then exit;',
  3495. 'end;',
  3496. 'begin',
  3497. ' Func1([1,2,3])']);
  3498. ConvertProgram;
  3499. CheckSource('TestFunctionResultInForLoop',
  3500. LinesToStr([ // statements
  3501. 'this.Func1 = function (a) {',
  3502. ' var Result = 0;',
  3503. ' for (var $l = rtl.length(a) - 1; $l >= 0; $l--) {',
  3504. ' Result = $l;',
  3505. ' if (a[Result] === 0) return Result;',
  3506. ' };',
  3507. ' for (var $in = a, $l1 = 0, $end = rtl.length($in) - 1; $l1 <= $end; $l1++) {',
  3508. ' Result = $in[$l1];',
  3509. ' if (a[Result] === 0) return Result;',
  3510. ' };',
  3511. ' return Result;',
  3512. '};',
  3513. '']),
  3514. LinesToStr([
  3515. '$mod.Func1([1, 2, 3]);'
  3516. ]));
  3517. end;
  3518. procedure TTestModule.TestFunctionResultInTypeCast;
  3519. begin
  3520. StartProgram(false);
  3521. Add([
  3522. 'function GetInt: longint;',
  3523. 'begin',
  3524. 'end;',
  3525. 'begin',
  3526. ' if Byte(GetInt)=0 then ;',
  3527. '']);
  3528. ConvertProgram;
  3529. CheckSource('TestFunctionResultInTypeCast',
  3530. LinesToStr([ // statements
  3531. 'this.GetInt = function () {',
  3532. ' var Result = 0;',
  3533. ' return Result;',
  3534. '};',
  3535. '']),
  3536. LinesToStr([
  3537. 'if (($mod.GetInt() & 255) === 0) ;'
  3538. ]));
  3539. end;
  3540. procedure TTestModule.TestExit;
  3541. begin
  3542. StartProgram(false);
  3543. Add('procedure ProcA;');
  3544. Add('begin');
  3545. Add(' exit;');
  3546. Add('end;');
  3547. Add('function FuncB: longint;');
  3548. Add('begin');
  3549. Add(' exit;');
  3550. Add(' exit(3);');
  3551. Add('end;');
  3552. Add('function FuncC: string;');
  3553. Add('begin');
  3554. Add(' exit;');
  3555. Add(' exit(''a'');');
  3556. Add(' exit(''abc'');');
  3557. Add('end;');
  3558. Add('begin');
  3559. Add(' exit;');
  3560. Add(' exit(1);');
  3561. ConvertProgram;
  3562. CheckSource('TestExit',
  3563. LinesToStr([ // statements
  3564. 'this.ProcA = function () {',
  3565. ' return;',
  3566. '};',
  3567. 'this.FuncB = function () {',
  3568. ' var Result = 0;',
  3569. ' return Result;',
  3570. ' return 3;',
  3571. ' return Result;',
  3572. '};',
  3573. 'this.FuncC = function () {',
  3574. ' var Result = "";',
  3575. ' return Result;',
  3576. ' return "a";',
  3577. ' return "abc";',
  3578. ' return Result;',
  3579. '};'
  3580. ]),
  3581. LinesToStr([
  3582. 'return;',
  3583. 'return 1;',
  3584. '']));
  3585. end;
  3586. procedure TTestModule.TestExit_ResultInFinally;
  3587. begin
  3588. StartProgram(false);
  3589. Add([
  3590. 'function Run: word;',
  3591. 'begin',
  3592. ' try',
  3593. ' exit(3);', // no Result in finally -> use return 3
  3594. ' finally',
  3595. ' end;',
  3596. 'end;',
  3597. 'function Fly: word;',
  3598. 'begin',
  3599. ' try',
  3600. ' exit(3);',
  3601. ' finally',
  3602. ' if Result>0 then ;',
  3603. ' end;',
  3604. 'end;',
  3605. 'function Jump: word;',
  3606. 'begin',
  3607. ' try',
  3608. ' try',
  3609. ' exit(4);',
  3610. ' finally',
  3611. ' end;',
  3612. ' finally',
  3613. ' if Result>0 then ;',
  3614. ' end;',
  3615. 'end;',
  3616. 'begin',
  3617. '']);
  3618. ConvertProgram;
  3619. CheckSource('TestExit_ResultInFinally',
  3620. LinesToStr([ // statements
  3621. 'this.Run = function () {',
  3622. ' var Result = 0;',
  3623. ' try {',
  3624. ' return 3;',
  3625. ' } finally {',
  3626. ' };',
  3627. ' return Result;',
  3628. '};',
  3629. 'this.Fly = function () {',
  3630. ' var Result = 0;',
  3631. ' try {',
  3632. ' Result = 3;',
  3633. ' return Result;',
  3634. ' } finally {',
  3635. ' if (Result > 0) ;',
  3636. ' };',
  3637. ' return Result;',
  3638. '};',
  3639. 'this.Jump = function () {',
  3640. ' var Result = 0;',
  3641. ' try {',
  3642. ' try {',
  3643. ' Result = 4;',
  3644. ' return Result;',
  3645. ' } finally {',
  3646. ' };',
  3647. ' } finally {',
  3648. ' if (Result > 0) ;',
  3649. ' };',
  3650. ' return Result;',
  3651. '};',
  3652. '']),
  3653. LinesToStr([
  3654. '']));
  3655. end;
  3656. procedure TTestModule.TestBreak;
  3657. begin
  3658. StartProgram(false);
  3659. Add([
  3660. 'var',
  3661. ' i: longint;',
  3662. 'begin',
  3663. ' repeat',
  3664. ' break;',
  3665. ' until true;',
  3666. ' while true do',
  3667. ' break;',
  3668. ' for i:=1 to 2 do',
  3669. ' break;']);
  3670. ConvertProgram;
  3671. CheckSource('TestBreak',
  3672. LinesToStr([ // statements
  3673. 'this.i = 0;'
  3674. ]),
  3675. LinesToStr([
  3676. 'do {',
  3677. ' break;',
  3678. '} while (!true);',
  3679. 'while (true) break;',
  3680. 'for ($mod.i = 1; $mod.i <= 2; $mod.i++) break;',
  3681. '']));
  3682. end;
  3683. procedure TTestModule.TestBreakAsVar;
  3684. begin
  3685. StartProgram(false);
  3686. Add([
  3687. 'procedure DoIt(break: boolean);',
  3688. 'begin',
  3689. ' if break then ;',
  3690. 'end;',
  3691. 'var',
  3692. ' break: boolean;',
  3693. 'begin',
  3694. ' if break then ;']);
  3695. ConvertProgram;
  3696. CheckSource('TestBreakAsVar',
  3697. LinesToStr([ // statements
  3698. 'this.DoIt = function (Break) {',
  3699. ' if (Break) ;',
  3700. '};',
  3701. 'this.Break = false;',
  3702. '']),
  3703. LinesToStr([
  3704. 'if($mod.Break) ;',
  3705. '']));
  3706. end;
  3707. procedure TTestModule.TestContinue;
  3708. begin
  3709. StartProgram(false);
  3710. Add('var i: longint;');
  3711. Add('begin');
  3712. Add(' repeat');
  3713. Add(' continue;');
  3714. Add(' until true;');
  3715. Add(' while true do');
  3716. Add(' continue;');
  3717. Add(' for i:=1 to 2 do');
  3718. Add(' continue;');
  3719. ConvertProgram;
  3720. CheckSource('TestContinue',
  3721. LinesToStr([ // statements
  3722. 'this.i = 0;'
  3723. ]),
  3724. LinesToStr([
  3725. 'do {',
  3726. ' continue;',
  3727. '} while (!true);',
  3728. 'while (true) continue;',
  3729. 'for ($mod.i = 1; $mod.i <= 2; $mod.i++) continue;',
  3730. '']));
  3731. end;
  3732. procedure TTestModule.TestProc_External;
  3733. begin
  3734. StartProgram(false);
  3735. Add('procedure Foo; external name ''console.log'';');
  3736. Add('function Bar: longint; external name ''get.item'';');
  3737. Add('function Bla(s: string): longint; external name ''apply.something'';');
  3738. Add('var');
  3739. Add(' i: longint;');
  3740. Add('begin');
  3741. Add(' Foo;');
  3742. Add(' i:=Bar;');
  3743. Add(' i:=Bla(''abc'');');
  3744. ConvertProgram;
  3745. CheckSource('TestProc_External',
  3746. LinesToStr([ // statements
  3747. 'this.i = 0;'
  3748. ]),
  3749. LinesToStr([
  3750. 'console.log();',
  3751. '$mod.i = get.item();',
  3752. '$mod.i = apply.something("abc");'
  3753. ]));
  3754. end;
  3755. procedure TTestModule.TestProc_ExternalOtherUnit;
  3756. begin
  3757. AddModuleWithIntfImplSrc('unit2.pas',
  3758. LinesToStr([
  3759. 'procedure Now; external name ''Date.now'';',
  3760. 'procedure DoIt;'
  3761. ]),
  3762. 'procedure doit; begin end;');
  3763. StartUnit(true);
  3764. Add('interface');
  3765. Add('uses unit2;');
  3766. Add('implementation');
  3767. Add('begin');
  3768. Add(' now;');
  3769. Add(' now();');
  3770. Add(' uNit2.now;');
  3771. Add(' uNit2.now();');
  3772. Add(' doit;');
  3773. Add(' uNit2.doit;');
  3774. ConvertUnit;
  3775. CheckSource('TestProc_ExternalOtherUnit',
  3776. LinesToStr([
  3777. '']),
  3778. LinesToStr([
  3779. 'Date.now();',
  3780. 'Date.now();',
  3781. 'Date.now();',
  3782. 'Date.now();',
  3783. 'pas.unit2.DoIt();',
  3784. 'pas.unit2.DoIt();',
  3785. '']));
  3786. end;
  3787. procedure TTestModule.TestProc_Asm;
  3788. begin
  3789. StartProgram(false);
  3790. Add([
  3791. '{$mode delphi}',
  3792. 'function DoIt: longint;',
  3793. 'begin;',
  3794. ' asm',
  3795. ' { a:{ b:{}, c:[]}, d:''1'' };',
  3796. ' end;',
  3797. ' asm console.log(); end;',
  3798. ' asm',
  3799. ' s = "'' ";',
  3800. ' s = ''" '';',
  3801. ' s = s + "world" + "''";',
  3802. ' // end',
  3803. ' s = ''end'';',
  3804. ' s = "end";',
  3805. ' s = "foo\"bar";',
  3806. ' s = ''a\''b'';',
  3807. ' s = `${expr}\`-"-''-`;',
  3808. ' s = `multi',
  3809. 'line`;',
  3810. ' end;',
  3811. 'end;',
  3812. 'procedure Fly;',
  3813. 'asm',
  3814. ' return;',
  3815. 'end;',
  3816. 'begin']);
  3817. ConvertProgram;
  3818. CheckSource('TestProc_Asm',
  3819. LinesToStr([ // statements
  3820. 'this.DoIt = function () {',
  3821. ' var Result = 0;',
  3822. ' { a:{ b:{}, c:[]}, d:''1'' };',
  3823. ' console.log();',
  3824. ' s = "'' ";',
  3825. ' s = ''" '';',
  3826. ' s = s + "world" + "''";',
  3827. ' // end',
  3828. ' s = ''end'';',
  3829. ' s = "end";',
  3830. ' s = "foo\"bar";',
  3831. ' s = ''a\''b'';',
  3832. ' s = `${expr}\`-"-''-`;',
  3833. ' s = `multi',
  3834. 'line`;',
  3835. ' return Result;',
  3836. '};',
  3837. 'this.Fly = function () {',
  3838. ' return;',
  3839. '};',
  3840. '']),
  3841. LinesToStr([
  3842. ''
  3843. ]));
  3844. end;
  3845. procedure TTestModule.TestProc_Assembler;
  3846. begin
  3847. StartProgram(false);
  3848. Add('function DoIt: longint; assembler;');
  3849. Add('asm');
  3850. Add('{ a:{ b:{}, c:[]}, d:''1'' };');
  3851. Add('end;');
  3852. Add('begin');
  3853. ConvertProgram;
  3854. CheckSource('TestProc_Assembler',
  3855. LinesToStr([ // statements
  3856. 'this.DoIt = function () {',
  3857. ' { a:{ b:{}, c:[]}, d:''1'' };',
  3858. '};'
  3859. ]),
  3860. LinesToStr([
  3861. ''
  3862. ]));
  3863. end;
  3864. procedure TTestModule.TestProc_VarParam;
  3865. begin
  3866. StartProgram(false);
  3867. Add('type integer = longint;');
  3868. Add('procedure DoIt(vG: integer; const vH: integer; var vI: integer);');
  3869. Add('var vJ: integer;');
  3870. Add('begin');
  3871. Add(' vg:=vg+1;');
  3872. Add(' vj:=vh+2;');
  3873. Add(' vi:=vi+3;');
  3874. Add(' doit(vg,vg,vg);');
  3875. Add(' doit(vh,vh,vj);');
  3876. Add(' doit(vi,vi,vi);');
  3877. Add(' doit(vj,vj,vj);');
  3878. Add('end;');
  3879. Add('var i: integer;');
  3880. Add('begin');
  3881. Add(' doit(i,i,i);');
  3882. ConvertProgram;
  3883. CheckSource('TestProc_VarParam',
  3884. LinesToStr([ // statements
  3885. 'this.DoIt = function (vG,vH,vI) {',
  3886. ' var vJ = 0;',
  3887. ' vG = vG + 1;',
  3888. ' vJ = vH + 2;',
  3889. ' vI.set(vI.get()+3);',
  3890. ' $mod.DoIt(vG, vG, {',
  3891. ' get: function () {',
  3892. ' return vG;',
  3893. ' },',
  3894. ' set: function (v) {',
  3895. ' vG = v;',
  3896. ' }',
  3897. ' });',
  3898. ' $mod.DoIt(vH, vH, {',
  3899. ' get: function () {',
  3900. ' return vJ;',
  3901. ' },',
  3902. ' set: function (v) {',
  3903. ' vJ = v;',
  3904. ' }',
  3905. ' });',
  3906. ' $mod.DoIt(vI.get(), vI.get(), vI);',
  3907. ' $mod.DoIt(vJ, vJ, {',
  3908. ' get: function () {',
  3909. ' return vJ;',
  3910. ' },',
  3911. ' set: function (v) {',
  3912. ' vJ = v;',
  3913. ' }',
  3914. ' });',
  3915. '};',
  3916. 'this.i = 0;'
  3917. ]),
  3918. LinesToStr([
  3919. '$mod.DoIt($mod.i,$mod.i,{',
  3920. ' p: $mod,',
  3921. ' get: function () {',
  3922. ' return this.p.i;',
  3923. ' },',
  3924. ' set: function (v) {',
  3925. ' this.p.i = v;',
  3926. ' }',
  3927. '});'
  3928. ]));
  3929. end;
  3930. procedure TTestModule.TestProc_VarParamString;
  3931. begin
  3932. StartProgram(false);
  3933. Add(['type TCaption = string;',
  3934. 'procedure DoIt(vA: TCaption; var vB: TCaption; out vC: TCaption);',
  3935. 'var c: char;',
  3936. 'begin',
  3937. ' va[1]:=c;',
  3938. ' vb[2]:=c;',
  3939. ' vc[3]:=c;',
  3940. 'end;',
  3941. 'begin']);
  3942. ConvertProgram;
  3943. CheckSource('TestProc_VarParamString',
  3944. LinesToStr([ // statements
  3945. 'this.DoIt = function (vA,vB,vC) {',
  3946. ' var c = "";',
  3947. ' vA = rtl.setCharAt(vA, 0, c);',
  3948. ' vB.set(rtl.setCharAt(vB.get(), 1, c));',
  3949. ' vC.set(rtl.setCharAt(vC.get(), 2, c));',
  3950. '};',
  3951. '']),
  3952. LinesToStr([
  3953. ]));
  3954. end;
  3955. procedure TTestModule.TestProc_VarParamV;
  3956. begin
  3957. StartProgram(false);
  3958. Add([
  3959. 'procedure Inc2(var i: longint);',
  3960. 'begin',
  3961. ' i:=i+2;',
  3962. 'end;',
  3963. 'procedure DoIt(v: longint);',
  3964. 'var p: array of longint;',
  3965. 'begin',
  3966. ' Inc2(v);',
  3967. ' Inc2(p[v]);',
  3968. 'end;',
  3969. 'begin']);
  3970. ConvertProgram;
  3971. CheckSource('TestProc_VarParamV',
  3972. LinesToStr([ // statements
  3973. 'this.Inc2 = function (i) {',
  3974. ' i.set(i.get()+2);',
  3975. '};',
  3976. 'this.DoIt = function (v) {',
  3977. ' var p = [];',
  3978. ' $mod.Inc2({get: function () {',
  3979. ' return v;',
  3980. ' }, set: function (w) {',
  3981. ' v = w;',
  3982. ' }});',
  3983. ' $mod.Inc2({',
  3984. ' a: v,',
  3985. ' p: p,',
  3986. ' get: function () {',
  3987. ' return this.p[this.a];',
  3988. ' },',
  3989. ' set: function (v) {',
  3990. ' this.p[this.a] = v;',
  3991. ' }',
  3992. ' });',
  3993. '};',
  3994. '']),
  3995. LinesToStr([
  3996. '']));
  3997. end;
  3998. procedure TTestModule.TestProc_Overload;
  3999. begin
  4000. StartProgram(false);
  4001. Add('procedure DoIt(vI: longint); begin end;');
  4002. Add('procedure DoIt(vI, vJ: longint); begin end;');
  4003. Add('procedure DoIt(vD: double); begin end;');
  4004. Add('begin');
  4005. Add(' DoIt(1);');
  4006. Add(' DoIt(2,3);');
  4007. Add(' DoIt(4.5);');
  4008. ConvertProgram;
  4009. CheckSource('TestProcedureOverload',
  4010. LinesToStr([ // statements
  4011. 'this.DoIt = function (vI) {',
  4012. '};',
  4013. 'this.DoIt$1 = function (vI, vJ) {',
  4014. '};',
  4015. 'this.DoIt$2 = function (vD) {',
  4016. '};',
  4017. '']),
  4018. LinesToStr([
  4019. '$mod.DoIt(1);',
  4020. '$mod.DoIt$1(2, 3);',
  4021. '$mod.DoIt$2(4.5);',
  4022. '']));
  4023. end;
  4024. procedure TTestModule.TestProc_OverloadForward;
  4025. begin
  4026. StartProgram(false);
  4027. Add('procedure DoIt(vI: longint); forward;');
  4028. Add('procedure DoIt(vI, vJ: longint); begin end;');
  4029. Add('procedure doit(vi: longint); begin end;');
  4030. Add('begin');
  4031. Add(' doit(1);');
  4032. Add(' doit(2,3);');
  4033. ConvertProgram;
  4034. CheckSource('TestProcedureOverloadForward',
  4035. LinesToStr([ // statements
  4036. 'this.DoIt$1 = function (vI, vJ) {',
  4037. '};',
  4038. 'this.DoIt = function (vI) {',
  4039. '};',
  4040. '']),
  4041. LinesToStr([
  4042. '$mod.DoIt(1);',
  4043. '$mod.DoIt$1(2, 3);',
  4044. '']));
  4045. end;
  4046. procedure TTestModule.TestProc_OverloadIntfImpl;
  4047. begin
  4048. StartUnit(false);
  4049. Add('interface');
  4050. Add('procedure DoIt(vI: longint);');
  4051. Add('procedure DoIt(vI, vJ: longint);');
  4052. Add('implementation');
  4053. Add('procedure DoIt(vI, vJ, vK, vL, vM: longint); forward;');
  4054. Add('procedure DoIt(vI, vJ, vK: longint); begin end;');
  4055. Add('procedure DoIt(vi: longint); begin end;');
  4056. Add('procedure DoIt(vI, vJ, vK, vL: longint); begin end;');
  4057. Add('procedure DoIt(vi, vj: longint); begin end;');
  4058. Add('procedure DoIt(vi, vj, vk, vl, vm: longint); begin end;');
  4059. Add('begin');
  4060. Add(' doit(1);');
  4061. Add(' doit(2,3);');
  4062. Add(' doit(4,5,6);');
  4063. Add(' doit(7,8,9,10);');
  4064. Add(' doit(11,12,13,14,15);');
  4065. ConvertUnit;
  4066. CheckSource('TestProcedureOverloadUnit',
  4067. LinesToStr([ // statements
  4068. 'var $impl = $mod.$impl;',
  4069. 'this.DoIt = function (vI) {',
  4070. '};',
  4071. 'this.DoIt$1 = function (vI, vJ) {',
  4072. '};',
  4073. '']),
  4074. LinesToStr([ // this.$init
  4075. '$mod.DoIt(1);',
  4076. '$mod.DoIt$1(2, 3);',
  4077. '$impl.DoIt$3(4,5,6);',
  4078. '$impl.DoIt$4(7,8,9,10);',
  4079. '$impl.DoIt$2(11,12,13,14,15);',
  4080. '']),
  4081. LinesToStr([ // implementation
  4082. '$impl.DoIt$3 = function (vI, vJ, vK) {',
  4083. '};',
  4084. '$impl.DoIt$4 = function (vI, vJ, vK, vL) {',
  4085. '};',
  4086. '$impl.DoIt$2 = function (vI, vJ, vK, vL, vM) {',
  4087. '};',
  4088. '']));
  4089. end;
  4090. procedure TTestModule.TestProc_OverloadNested;
  4091. begin
  4092. StartProgram(false);
  4093. Add([
  4094. 'procedure doit(vA: longint);',
  4095. ' procedure DoIt(vA, vB: longint); overload;',
  4096. ' begin',
  4097. ' doit(1);',
  4098. ' doit(1,2);',
  4099. ' end;',
  4100. ' procedure doit(vA, vB, vC: longint);',
  4101. ' begin',
  4102. ' doit(1);',
  4103. ' doit(1,2);',
  4104. ' doit(1,2,3);',
  4105. ' end;',
  4106. 'begin',
  4107. ' doit(1);',
  4108. ' doit(1,2);',
  4109. ' doit(1,2,3);',
  4110. 'end;',
  4111. 'begin // main',
  4112. ' doit(1);']);
  4113. ConvertProgram;
  4114. CheckSource('TestProcedureOverloadNested',
  4115. LinesToStr([ // statements
  4116. 'this.doit = function (vA) {',
  4117. ' function DoIt$1(vA, vB) {',
  4118. ' $mod.doit(1);',
  4119. ' DoIt$1(1, 2);',
  4120. ' };',
  4121. ' function doit$2(vA, vB, vC) {',
  4122. ' $mod.doit(1);',
  4123. ' DoIt$1(1, 2);',
  4124. ' doit$2(1, 2, 3);',
  4125. ' };',
  4126. ' $mod.doit(1);',
  4127. ' DoIt$1(1, 2);',
  4128. ' doit$2(1, 2, 3);',
  4129. '};',
  4130. '']),
  4131. LinesToStr([
  4132. '$mod.doit(1);',
  4133. '']));
  4134. end;
  4135. procedure TTestModule.TestProc_OverloadNestedForward;
  4136. begin
  4137. StartProgram(false);
  4138. Add([
  4139. 'procedure DoIt(vA: longint); overload; forward;',
  4140. 'procedure DoIt(vB, vC: longint); overload;',
  4141. 'begin // 2 param overload',
  4142. ' doit(1);',
  4143. ' doit(1,2);',
  4144. 'end;',
  4145. 'procedure doit(vA: longint);',
  4146. ' procedure DoIt(vA, vB, vC: longint); overload; forward;',
  4147. ' procedure DoIt(vA, vB, vC, vD: longint); overload;',
  4148. ' begin // 4 param overload',
  4149. ' doit(1);',
  4150. ' doit(1,2);',
  4151. ' doit(1,2,3);',
  4152. ' doit(1,2,3,4);',
  4153. ' end;',
  4154. ' procedure doit(vA, vB, vC: longint);',
  4155. ' procedure DoIt(vA, vB, vC, vD, vE: longint); overload; forward;',
  4156. ' procedure DoIt(vA, vB, vC, vD, vE, vF: longint); overload;',
  4157. ' begin // 6 param overload',
  4158. ' doit(1);',
  4159. ' doit(1,2);',
  4160. ' doit(1,2,3);',
  4161. ' doit(1,2,3,4);',
  4162. ' doit(1,2,3,4,5);',
  4163. ' doit(1,2,3,4,5,6);',
  4164. ' end;',
  4165. ' procedure doit(vA, vB, vC, vD, vE: longint);',
  4166. ' begin // 5 param overload',
  4167. ' doit(1);',
  4168. ' doit(1,2);',
  4169. ' doit(1,2,3);',
  4170. ' doit(1,2,3,4);',
  4171. ' doit(1,2,3,4,5);',
  4172. ' doit(1,2,3,4,5,6);',
  4173. ' end;',
  4174. ' begin // 3 param overload',
  4175. ' doit(1);',
  4176. ' doit(1,2);',
  4177. ' doit(1,2,3);',
  4178. ' doit(1,2,3,4);',
  4179. ' doit(1,2,3,4,5);',
  4180. ' doit(1,2,3,4,5,6);',
  4181. ' end;',
  4182. 'begin // 1 param overload',
  4183. ' doit(1);',
  4184. ' doit(1,2);',
  4185. ' doit(1,2,3);',
  4186. ' doit(1,2,3,4);',
  4187. 'end;',
  4188. 'begin // main',
  4189. ' doit(1);',
  4190. ' doit(1,2);']);
  4191. ConvertProgram;
  4192. CheckSource('TestProc_OverloadNestedForward',
  4193. LinesToStr([ // statements
  4194. 'this.DoIt$1 = function (vB, vC) {',
  4195. ' $mod.DoIt(1);',
  4196. ' $mod.DoIt$1(1, 2);',
  4197. '};',
  4198. 'this.DoIt = function (vA) {',
  4199. ' function DoIt$3(vA, vB, vC, vD) {',
  4200. ' $mod.DoIt(1);',
  4201. ' $mod.DoIt$1(1, 2);',
  4202. ' DoIt$2(1, 2, 3);',
  4203. ' DoIt$3(1, 2, 3, 4);',
  4204. ' };',
  4205. ' function DoIt$2(vA, vB, vC) {',
  4206. ' function DoIt$5(vA, vB, vC, vD, vE, vF) {',
  4207. ' $mod.DoIt(1);',
  4208. ' $mod.DoIt$1(1, 2);',
  4209. ' DoIt$2(1, 2, 3);',
  4210. ' DoIt$3(1, 2, 3, 4);',
  4211. ' DoIt$4(1, 2, 3, 4, 5);',
  4212. ' DoIt$5(1, 2, 3, 4, 5, 6);',
  4213. ' };',
  4214. ' function DoIt$4(vA, vB, vC, vD, vE) {',
  4215. ' $mod.DoIt(1);',
  4216. ' $mod.DoIt$1(1, 2);',
  4217. ' DoIt$2(1, 2, 3);',
  4218. ' DoIt$3(1, 2, 3, 4);',
  4219. ' DoIt$4(1, 2, 3, 4, 5);',
  4220. ' DoIt$5(1, 2, 3, 4, 5, 6);',
  4221. ' };',
  4222. ' $mod.DoIt(1);',
  4223. ' $mod.DoIt$1(1, 2);',
  4224. ' DoIt$2(1, 2, 3);',
  4225. ' DoIt$3(1, 2, 3, 4);',
  4226. ' DoIt$4(1, 2, 3, 4, 5);',
  4227. ' DoIt$5(1, 2, 3, 4, 5, 6);',
  4228. ' };',
  4229. ' $mod.DoIt(1);',
  4230. ' $mod.DoIt$1(1, 2);',
  4231. ' DoIt$2(1, 2, 3);',
  4232. ' DoIt$3(1, 2, 3, 4);',
  4233. '};',
  4234. '']),
  4235. LinesToStr([
  4236. '$mod.DoIt(1);',
  4237. '$mod.DoIt$1(1, 2);',
  4238. '']));
  4239. end;
  4240. procedure TTestModule.TestProc_OverloadUnitCycle;
  4241. begin
  4242. AddModuleWithIntfImplSrc('Unit2.pas',
  4243. LinesToStr([
  4244. 'type',
  4245. ' TObject = class',
  4246. ' procedure DoIt(b: boolean); virtual; abstract;',
  4247. ' procedure DoIt(i: longint); virtual; abstract;',
  4248. ' end;',
  4249. '']),
  4250. 'uses test1;');
  4251. StartUnit(true);
  4252. Add([
  4253. 'interface',
  4254. 'uses unit2;',
  4255. 'type',
  4256. ' TEagle = class(TObject)',
  4257. ' procedure DoIt(b: boolean); override;',
  4258. ' procedure DoIt(i: longint); override;',
  4259. ' end;',
  4260. 'implementation',
  4261. 'procedure TEagle.DoIt(b: boolean); begin end;',
  4262. 'procedure TEagle.DoIt(i: longint); begin end;',
  4263. '']);
  4264. ConvertUnit;
  4265. CheckSource('TestProc_OverloadUnitCycle',
  4266. LinesToStr([ // statements
  4267. 'rtl.createClass(this, "TEagle", pas.Unit2.TObject, function () {',
  4268. ' this.DoIt = function (b) {',
  4269. ' };',
  4270. ' this.DoIt$1 = function (i) {',
  4271. ' };',
  4272. '});',
  4273. '']),
  4274. '',
  4275. LinesToStr([
  4276. '']));
  4277. end;
  4278. procedure TTestModule.TestProc_Varargs;
  4279. begin
  4280. StartProgram(false);
  4281. Add([
  4282. 'procedure ProcA(i:longint); varargs; external name ''ProcA'';',
  4283. 'procedure ProcB; varargs; external name ''ProcB'';',
  4284. 'procedure ProcC(i: longint = 17); varargs; external name ''ProcC'';',
  4285. 'function GetIt: longint; begin end;',
  4286. 'begin',
  4287. ' ProcA(1);',
  4288. ' ProcA(1,2);',
  4289. ' ProcA(1,2.0);',
  4290. ' ProcA(1,2,3);',
  4291. ' ProcA(1,''2'');',
  4292. ' ProcA(2,'''');',
  4293. ' ProcA(3,false);',
  4294. ' ProcB;',
  4295. ' ProcB();',
  4296. ' ProcB(4);',
  4297. ' ProcB(''foo'');',
  4298. ' ProcC;',
  4299. ' ProcC();',
  4300. ' ProcC(4);',
  4301. ' ProcC(5,''foo'');',
  4302. ' ProcB(GetIt);',
  4303. ' ProcB(GetIt());',
  4304. ' ProcB(GetIt,GetIt());']);
  4305. ConvertProgram;
  4306. CheckSource('TestProc_Varargs',
  4307. LinesToStr([ // statements
  4308. 'this.GetIt = function () {',
  4309. ' var Result = 0;',
  4310. ' return Result;',
  4311. '};',
  4312. '']),
  4313. LinesToStr([
  4314. 'ProcA(1);',
  4315. 'ProcA(1, 2);',
  4316. 'ProcA(1, 2.0);',
  4317. 'ProcA(1, 2, 3);',
  4318. 'ProcA(1, "2");',
  4319. 'ProcA(2, "");',
  4320. 'ProcA(3, false);',
  4321. 'ProcB();',
  4322. 'ProcB();',
  4323. 'ProcB(4);',
  4324. 'ProcB("foo");',
  4325. 'ProcC(17);',
  4326. 'ProcC(17);',
  4327. 'ProcC(4);',
  4328. 'ProcC(5, "foo");',
  4329. 'ProcB($mod.GetIt());',
  4330. 'ProcB($mod.GetIt());',
  4331. 'ProcB($mod.GetIt(), $mod.GetIt());',
  4332. '']));
  4333. end;
  4334. procedure TTestModule.TestProc_ConstOrder;
  4335. begin
  4336. StartProgram(false);
  4337. Add([
  4338. 'const A = 3;',
  4339. 'const B = A+1;',
  4340. 'procedure DoIt;',
  4341. 'const C = A+1;',
  4342. 'const D = B+1;',
  4343. 'const E = D+C+B+A;',
  4344. 'begin',
  4345. 'end;',
  4346. 'begin'
  4347. ]);
  4348. ConvertProgram;
  4349. CheckSource('TestProc_ConstOrder',
  4350. LinesToStr([ // statements
  4351. 'this.A = 3;',
  4352. 'this.B = 3 + 1;',
  4353. 'var C = 3 + 1;',
  4354. 'var D = 4 + 1;',
  4355. 'var E = 5 + 4 + 4 + 3;',
  4356. 'this.DoIt = function () {',
  4357. '};',
  4358. '']),
  4359. LinesToStr([
  4360. ''
  4361. ]));
  4362. end;
  4363. procedure TTestModule.TestProc_DuplicateConst;
  4364. begin
  4365. StartProgram(false);
  4366. Add([
  4367. 'const A = 1;',
  4368. 'procedure DoIt;',
  4369. 'const A = 2;',
  4370. ' procedure SubIt;',
  4371. ' const A = 21;',
  4372. ' begin',
  4373. ' end;',
  4374. 'begin',
  4375. 'end;',
  4376. 'procedure DoSome;',
  4377. 'const A = 3;',
  4378. 'begin',
  4379. 'end;',
  4380. 'begin'
  4381. ]);
  4382. ConvertProgram;
  4383. CheckSource('TestProc_DuplicateConst',
  4384. LinesToStr([ // statements
  4385. 'this.A = 1;',
  4386. 'var A$1 = 2;',
  4387. 'var A$2 = 21;',
  4388. 'this.DoIt = function () {',
  4389. ' function SubIt() {',
  4390. ' };',
  4391. '};',
  4392. 'var A$3 = 3;',
  4393. 'this.DoSome = function () {',
  4394. '};',
  4395. '']),
  4396. LinesToStr([
  4397. ''
  4398. ]));
  4399. end;
  4400. procedure TTestModule.TestProc_LocalVarAbsolute;
  4401. begin
  4402. StartProgram(false);
  4403. Add([
  4404. 'type',
  4405. ' TObject = class',
  4406. ' Index: longint;',
  4407. ' procedure DoAbs(Item: pointer);',
  4408. ' end;',
  4409. 'procedure TObject.DoAbs(Item: pointer);',
  4410. 'var',
  4411. ' o: TObject absolute Item;',
  4412. 'begin',
  4413. ' if o.Index<o.Index then o.Index:=o.Index;',
  4414. 'end;',
  4415. 'procedure DoIt(i: longint; p: pointer);',
  4416. 'var',
  4417. ' d: double absolute i;',
  4418. ' s: string absolute d;',
  4419. ' oi: TObject absolute i;',
  4420. ' op: TObject absolute p;',
  4421. 'begin',
  4422. ' if d=d then d:=d;',
  4423. ' if s=s then s:=s;',
  4424. ' if oi.Index<oi.Index then oi.Index:=oi.Index;',
  4425. ' if op.Index=op.Index then op.Index:=op.Index;',
  4426. 'end;',
  4427. 'begin']);
  4428. ConvertProgram;
  4429. CheckSource('TestProc_LocalVarAbsolute',
  4430. LinesToStr([ // statements
  4431. 'rtl.createClass(this, "TObject", null, function () {',
  4432. ' this.$init = function () {',
  4433. ' this.Index = 0;',
  4434. ' };',
  4435. ' this.$final = function () {',
  4436. ' };',
  4437. ' this.DoAbs = function (Item) {',
  4438. ' if (Item.Index < Item.Index) Item.Index = Item.Index;',
  4439. ' };',
  4440. '});',
  4441. 'this.DoIt = function (i, p) {',
  4442. ' if (i === i) i = i;',
  4443. ' if (i === i) i = i;',
  4444. ' if (i.Index < i.Index) i.Index = i.Index;',
  4445. ' if (p.Index === p.Index) p.Index = p.Index;',
  4446. '};'
  4447. ]),
  4448. LinesToStr([
  4449. ]));
  4450. end;
  4451. procedure TTestModule.TestProc_LocalVarInit;
  4452. begin
  4453. StartProgram(false);
  4454. Add([
  4455. 'type TBytes = array of byte;',
  4456. 'procedure DoIt;',
  4457. 'const c = 4;',
  4458. 'var',
  4459. ' b: byte = 1;',
  4460. ' w: word = 2+c;',
  4461. ' p: pointer = nil;',
  4462. ' Buffer: TBytes = nil;',
  4463. 'begin',
  4464. 'end;',
  4465. 'begin']);
  4466. ConvertProgram;
  4467. CheckSource('TestProc_LocalVarInit',
  4468. LinesToStr([ // statements
  4469. 'var c = 4;',
  4470. 'this.DoIt = function () {',
  4471. ' var b = 1;',
  4472. ' var w = 2 + 4;',
  4473. ' var p = null;',
  4474. ' var Buffer = [];',
  4475. '};',
  4476. '']),
  4477. LinesToStr([
  4478. ]));
  4479. end;
  4480. procedure TTestModule.TestProc_ReservedWords;
  4481. begin
  4482. StartProgram(false);
  4483. Add([
  4484. 'procedure Date(ArrayBuffer: longint);',
  4485. 'const',
  4486. ' NaN: longint = 3;',
  4487. 'var',
  4488. ' &Boolean: longint;',
  4489. ' procedure Error(ArrayBuffer: longint);',
  4490. ' begin',
  4491. ' end;',
  4492. 'begin',
  4493. ' Nan:=&bOolean;',
  4494. 'end;',
  4495. 'begin',
  4496. ' Date(1);']);
  4497. ConvertProgram;
  4498. CheckSource('TestProc_ReservedWords',
  4499. LinesToStr([ // statements
  4500. 'var naN = 3;',
  4501. 'this.Date = function (arrayBuffer) {',
  4502. ' var boolean = 0;',
  4503. ' function error(arrayBuffer) {',
  4504. ' };',
  4505. ' naN = boolean;',
  4506. '};',
  4507. '']),
  4508. LinesToStr([
  4509. ' $mod.Date(1);'
  4510. ]));
  4511. end;
  4512. procedure TTestModule.TestProc_ConstRefWord;
  4513. begin
  4514. StartProgram(false);
  4515. Add([
  4516. 'procedure Run(constref w: word);',
  4517. 'var l: word;',
  4518. 'begin',
  4519. ' l:=w;',
  4520. ' Run(w);',
  4521. ' Run(l);',
  4522. 'end;',
  4523. 'procedure Fly(a: word; var b: word; out c: word; const d: word; constref e: word);',
  4524. 'begin',
  4525. ' Run(a);',
  4526. ' Run(b);',
  4527. ' Run(c);',
  4528. ' Run(d);',
  4529. ' Run(e);',
  4530. 'end;',
  4531. 'begin',
  4532. ' Run(1);']);
  4533. ConvertProgram;
  4534. CheckHint(mtWarning,nConstRefNotForXAsConst,'ConstRef not yet implemented for Word. Treating as Const');
  4535. CheckSource('TestProc_ConstRefWord',
  4536. LinesToStr([ // statements
  4537. 'this.Run = function (w) {',
  4538. ' var l = 0;',
  4539. ' l = w;',
  4540. ' $mod.Run(w);',
  4541. ' $mod.Run(l);',
  4542. '};',
  4543. 'this.Fly = function (a, b, c, d, e) {',
  4544. ' $mod.Run(a);',
  4545. ' $mod.Run(b.get());',
  4546. ' $mod.Run(c.get());',
  4547. ' $mod.Run(d);',
  4548. ' $mod.Run(e);',
  4549. '};',
  4550. '']),
  4551. LinesToStr([
  4552. '$mod.Run(1);'
  4553. ]));
  4554. end;
  4555. procedure TTestModule.TestAnonymousProc_Assign_ObjFPC;
  4556. begin
  4557. StartProgram(false);
  4558. Add([
  4559. '{$mode objfpc}',
  4560. 'type',
  4561. ' TFunc = reference to function(x: word): word;',
  4562. 'var Func: TFunc;',
  4563. 'procedure DoIt(a: word);',
  4564. 'begin',
  4565. ' Func:=function(b:word): word',
  4566. ' begin',
  4567. ' Result:=a+b;',
  4568. ' exit(b);',
  4569. ' exit(Result);',
  4570. ' end;',// test semicolon
  4571. ' a:=3;',
  4572. 'end;',
  4573. 'begin',
  4574. ' Func:=function(c:word):word begin',
  4575. ' Result:=3+c;',
  4576. ' exit(c);',
  4577. ' exit(Result);',
  4578. ' end;']);
  4579. ConvertProgram;
  4580. CheckSource('TestAnonymousProc_Assign_ObjFPC',
  4581. LinesToStr([ // statements
  4582. 'this.Func = null;',
  4583. 'this.DoIt = function (a) {',
  4584. ' $mod.Func = function (b) {',
  4585. ' var Result = 0;',
  4586. ' Result = a + b;',
  4587. ' return b;',
  4588. ' return Result;',
  4589. ' return Result;',
  4590. ' };',
  4591. ' a = 3;',
  4592. '};',
  4593. '']),
  4594. LinesToStr([
  4595. '$mod.Func = function (c) {',
  4596. ' var Result = 0;',
  4597. ' Result = 3 + c;',
  4598. ' return c;',
  4599. ' return Result;',
  4600. ' return Result;',
  4601. '};',
  4602. '']));
  4603. end;
  4604. procedure TTestModule.TestAnonymousProc_Assign_Delphi;
  4605. begin
  4606. StartProgram(false);
  4607. Add([
  4608. '{$mode delphi}',
  4609. 'type',
  4610. ' TProc = reference to procedure(x: word);',
  4611. 'procedure DoIt(a: word);',
  4612. 'var Proc: TProc;',
  4613. 'begin',
  4614. ' Proc:=procedure(b:word) begin end;',
  4615. 'end;',
  4616. 'var Proc: TProc;',
  4617. 'begin',
  4618. ' Proc:=procedure(c:word) begin end;',
  4619. '']);
  4620. ConvertProgram;
  4621. CheckSource('TestAnonymousProc_Assign_Delphi',
  4622. LinesToStr([ // statements
  4623. 'this.DoIt = function (a) {',
  4624. ' var Proc = null;',
  4625. ' Proc = function (b) {',
  4626. ' };',
  4627. '};',
  4628. 'this.Proc = null;',
  4629. '']),
  4630. LinesToStr([
  4631. '$mod.Proc = function (c) {',
  4632. '};',
  4633. '']));
  4634. end;
  4635. procedure TTestModule.TestAnonymousProc_Arg;
  4636. begin
  4637. StartProgram(false);
  4638. Add([
  4639. 'type',
  4640. ' TProc = reference to procedure;',
  4641. ' TFunc = reference to function(x: word): word;',
  4642. 'procedure DoMore(f,g: TProc);',
  4643. 'begin',
  4644. 'end;',
  4645. 'procedure DoOdd(v: jsvalue);',
  4646. 'begin',
  4647. 'end;',
  4648. 'procedure DoIt(f: TFunc);',
  4649. 'begin',
  4650. ' DoIt(function(b:word): word',
  4651. ' begin',
  4652. ' Result:=1+b;',
  4653. ' end);',
  4654. ' DoMore(procedure begin end, procedure begin end);',
  4655. ' DoOdd(procedure begin end);',
  4656. 'end;',
  4657. 'begin',
  4658. ' DoMore(procedure begin end,',
  4659. ' procedure assembler asm',
  4660. ' console.log("c");',
  4661. ' end);',
  4662. '']);
  4663. ConvertProgram;
  4664. CheckSource('TestAnonymousProc_Arg',
  4665. LinesToStr([ // statements
  4666. 'this.DoMore = function (f, g) {',
  4667. '};',
  4668. 'this.DoOdd = function (v) {',
  4669. '};',
  4670. 'this.DoIt = function (f) {',
  4671. ' $mod.DoIt(function (b) {',
  4672. ' var Result = 0;',
  4673. ' Result = 1 + b;',
  4674. ' return Result;',
  4675. ' });',
  4676. ' $mod.DoMore(function () {',
  4677. ' }, function () {',
  4678. ' });',
  4679. ' $mod.DoOdd(function () {',
  4680. ' });',
  4681. '};',
  4682. '']),
  4683. LinesToStr([
  4684. '$mod.DoMore(function () {',
  4685. '}, function () {',
  4686. ' console.log("c");',
  4687. '});',
  4688. '']));
  4689. end;
  4690. procedure TTestModule.TestAnonymousProc_Typecast;
  4691. begin
  4692. StartProgram(false);
  4693. Add([
  4694. 'type',
  4695. ' TProc = reference to procedure(w: word);',
  4696. ' TArr = array of word;',
  4697. ' TFuncArr = reference to function: TArr;',
  4698. 'procedure DoIt(p: TProc);',
  4699. 'var',
  4700. ' w: word;',
  4701. ' a: TArr;',
  4702. 'begin',
  4703. ' p:=TProc(procedure(b: smallint) begin end);',
  4704. ' a:=TFuncArr(function: TArr begin end)();',
  4705. ' w:=TFuncArr(function: TArr begin end)()[3];',
  4706. 'end;',
  4707. 'begin']);
  4708. ConvertProgram;
  4709. CheckSource('TestAnonymousProc_Typecast',
  4710. LinesToStr([ // statements
  4711. 'this.DoIt = function (p) {',
  4712. ' var w = 0;',
  4713. ' var a = [];',
  4714. ' p = function (b) {',
  4715. ' };',
  4716. ' a = function () {',
  4717. ' var Result = [];',
  4718. ' return Result;',
  4719. ' }();',
  4720. ' w = function () {',
  4721. ' var Result = [];',
  4722. ' return Result;',
  4723. ' }()[3];',
  4724. '};',
  4725. '']),
  4726. LinesToStr([
  4727. '']));
  4728. end;
  4729. procedure TTestModule.TestAnonymousProc_With;
  4730. begin
  4731. StartProgram(false);
  4732. Add([
  4733. 'type',
  4734. ' TProc = reference to procedure(w: word);',
  4735. ' TObject = class',
  4736. ' b: boolean;',
  4737. ' end;',
  4738. 'var',
  4739. ' p: TProc;',
  4740. ' bird: TObject;',
  4741. 'begin',
  4742. ' with bird do',
  4743. ' p:=procedure(w: word)',
  4744. ' begin',
  4745. ' b:=w>2;',
  4746. ' end;',
  4747. '']);
  4748. ConvertProgram;
  4749. CheckSource('TestAnonymousProc_With',
  4750. LinesToStr([ // statements
  4751. 'rtl.createClass(this, "TObject", null, function () {',
  4752. ' this.$init = function () {',
  4753. ' this.b = false;',
  4754. ' };',
  4755. ' this.$final = function () {',
  4756. ' };',
  4757. '});',
  4758. 'this.p = null;',
  4759. 'this.bird = null;',
  4760. '']),
  4761. LinesToStr([
  4762. 'var $with = $mod.bird;',
  4763. '$mod.p = function (w) {',
  4764. ' $with.b = w > 2;',
  4765. '};',
  4766. '']));
  4767. end;
  4768. procedure TTestModule.TestAnonymousProc_ExceptOn;
  4769. begin
  4770. StartProgram(false);
  4771. Add([
  4772. 'type',
  4773. ' TProc = reference to procedure;',
  4774. ' TObject = class',
  4775. ' b: boolean;',
  4776. ' end;',
  4777. 'procedure DoIt;',
  4778. 'var',
  4779. ' p: TProc;',
  4780. 'begin',
  4781. ' try',
  4782. ' except',
  4783. ' on E: TObject do',
  4784. ' p:=procedure',
  4785. ' begin',
  4786. ' E.b:=true;',
  4787. ' end;',
  4788. ' end;',
  4789. 'end;',
  4790. 'begin']);
  4791. ConvertProgram;
  4792. CheckSource('TestAnonymousProc_ExceptOn',
  4793. LinesToStr([ // statements
  4794. 'rtl.createClass(this, "TObject", null, function () {',
  4795. ' this.$init = function () {',
  4796. ' this.b = false;',
  4797. ' };',
  4798. ' this.$final = function () {',
  4799. ' };',
  4800. '});',
  4801. 'this.DoIt = function () {',
  4802. ' var p = null;',
  4803. ' try {} catch ($e) {',
  4804. ' if ($mod.TObject.isPrototypeOf($e)) {',
  4805. ' var E = $e;',
  4806. ' p = function () {',
  4807. ' E.b = true;',
  4808. ' };',
  4809. ' } else throw $e',
  4810. ' };',
  4811. '};',
  4812. '']),
  4813. LinesToStr([
  4814. '']));
  4815. end;
  4816. procedure TTestModule.TestAnonymousProc_Nested;
  4817. begin
  4818. StartProgram(false);
  4819. Add([
  4820. 'type',
  4821. ' TProc = reference to procedure;',
  4822. ' TObject = class',
  4823. ' i: byte;',
  4824. ' procedure DoIt;',
  4825. ' end;',
  4826. 'procedure TObject.DoIt;',
  4827. 'var',
  4828. ' p: TProc;',
  4829. ' procedure Sub;',
  4830. ' begin',
  4831. ' p:=procedure',
  4832. ' begin',
  4833. ' i:=3;',
  4834. ' Self.i:=4;',
  4835. ' p:=procedure',
  4836. ' procedure SubSub;',
  4837. ' begin',
  4838. ' i:=13;',
  4839. ' Self.i:=14;',
  4840. ' end;',
  4841. ' begin',
  4842. ' i:=13;',
  4843. ' Self.i:=14;',
  4844. ' end;',
  4845. ' end;',
  4846. ' end;',
  4847. 'begin',
  4848. 'end;',
  4849. 'begin']);
  4850. ConvertProgram;
  4851. CheckSource('TestAnonymousProc_Nested',
  4852. LinesToStr([ // statements
  4853. 'rtl.createClass(this, "TObject", null, function () {',
  4854. ' this.$init = function () {',
  4855. ' this.i = 0;',
  4856. ' };',
  4857. ' this.$final = function () {',
  4858. ' };',
  4859. ' this.DoIt = function () {',
  4860. ' var $Self = this;',
  4861. ' var p = null;',
  4862. ' function Sub() {',
  4863. ' p = function () {',
  4864. ' $Self.i = 3;',
  4865. ' $Self.i = 4;',
  4866. ' p = function () {',
  4867. ' function SubSub() {',
  4868. ' $Self.i = 13;',
  4869. ' $Self.i = 14;',
  4870. ' };',
  4871. ' $Self.i = 13;',
  4872. ' $Self.i = 14;',
  4873. ' };',
  4874. ' };',
  4875. ' };',
  4876. ' };',
  4877. '});',
  4878. '']),
  4879. LinesToStr([
  4880. '']));
  4881. end;
  4882. procedure TTestModule.TestAnonymousProc_NestedAssignResult;
  4883. begin
  4884. StartProgram(false);
  4885. Add([
  4886. 'type',
  4887. ' TProc = reference to procedure;',
  4888. 'function DoIt: TProc;',
  4889. ' function Sub: TProc;',
  4890. ' begin',
  4891. ' Result:=procedure',
  4892. ' begin',
  4893. ' Sub:=procedure',
  4894. ' procedure SubSub;',
  4895. ' begin',
  4896. ' Result:=nil;',
  4897. ' Sub:=nil;',
  4898. ' DoIt:=nil;',
  4899. ' end;',
  4900. ' begin',
  4901. ' Result:=nil;',
  4902. ' Sub:=nil;',
  4903. ' DoIt:=nil;',
  4904. ' end;',
  4905. ' end;',
  4906. ' end;',
  4907. 'begin',
  4908. 'end;',
  4909. 'begin']);
  4910. ConvertProgram;
  4911. CheckSource('TestAnonymousProc_NestedAssignResult',
  4912. LinesToStr([ // statements
  4913. 'this.DoIt = function () {',
  4914. ' var Result = null;',
  4915. ' function Sub() {',
  4916. ' var Result$1 = null;',
  4917. ' Result$1 = function () {',
  4918. ' Result$1 = function () {',
  4919. ' function SubSub() {',
  4920. ' Result$1 = null;',
  4921. ' Result$1 = null;',
  4922. ' Result = null;',
  4923. ' };',
  4924. ' Result$1 = null;',
  4925. ' Result$1 = null;',
  4926. ' Result = null;',
  4927. ' };',
  4928. ' };',
  4929. ' return Result$1;',
  4930. ' };',
  4931. ' return Result;',
  4932. '};',
  4933. '']),
  4934. LinesToStr([
  4935. '']));
  4936. end;
  4937. procedure TTestModule.TestAnonymousProc_Class;
  4938. begin
  4939. StartProgram(false);
  4940. Add([
  4941. 'type',
  4942. ' TProc = reference to procedure;',
  4943. ' TEvent = procedure of object;',
  4944. ' TObject = class',
  4945. ' Size: word;',
  4946. ' function GetIt: TProc;',
  4947. ' procedure DoIt; virtual; abstract;',
  4948. ' end;',
  4949. 'function TObject.GetIt: TProc;',
  4950. 'begin',
  4951. ' Result:=procedure',
  4952. ' var p: TEvent;',
  4953. ' begin',
  4954. ' Size:=Size;',
  4955. ' Size:=Self.Size;',
  4956. ' p:=@DoIt;',
  4957. ' p:[email protected];',
  4958. ' end;',
  4959. 'end;',
  4960. 'begin']);
  4961. ConvertProgram;
  4962. CheckSource('TestAnonymousProc_Class',
  4963. LinesToStr([ // statements
  4964. 'rtl.createClass(this, "TObject", null, function () {',
  4965. ' this.$init = function () {',
  4966. ' this.Size = 0;',
  4967. ' };',
  4968. ' this.$final = function () {',
  4969. ' };',
  4970. ' this.GetIt = function () {',
  4971. ' var $Self = this;',
  4972. ' var Result = null;',
  4973. ' Result = function () {',
  4974. ' var p = null;',
  4975. ' $Self.Size = $Self.Size;',
  4976. ' $Self.Size = $Self.Size;',
  4977. ' p = rtl.createCallback($Self, "DoIt");',
  4978. ' p = rtl.createCallback($Self, "DoIt");',
  4979. ' };',
  4980. ' return Result;',
  4981. ' };',
  4982. '});',
  4983. '']),
  4984. LinesToStr([
  4985. '']));
  4986. end;
  4987. procedure TTestModule.TestAnonymousProc_ForLoop;
  4988. begin
  4989. StartProgram(false);
  4990. Add([
  4991. 'type TProc = reference to procedure;',
  4992. 'procedure Foo(p: TProc);',
  4993. 'begin',
  4994. 'end;',
  4995. 'procedure DoIt;',
  4996. 'var i: word;',
  4997. ' a: word;',
  4998. 'begin',
  4999. ' for i:=1 to 10 do begin',
  5000. ' Foo(procedure begin a:=3; end);',
  5001. ' end;',
  5002. 'end;',
  5003. 'begin',
  5004. ' DoIt;']);
  5005. ConvertProgram;
  5006. CheckSource('TestAnonymousProc_ForLoop',
  5007. LinesToStr([ // statements
  5008. 'this.Foo = function (p) {',
  5009. '};',
  5010. 'this.DoIt = function () {',
  5011. ' var i = 0;',
  5012. ' var a = 0;',
  5013. ' for (i = 1; i <= 10; i++) {',
  5014. ' $mod.Foo(function () {',
  5015. ' a = 3;',
  5016. ' });',
  5017. ' };',
  5018. '};',
  5019. '']),
  5020. LinesToStr([
  5021. '$mod.DoIt();'
  5022. ]));
  5023. end;
  5024. procedure TTestModule.TestAnonymousProc_AsmDelphi;
  5025. begin
  5026. StartProgram(false);
  5027. Add([
  5028. '{$mode delphi}',
  5029. 'type',
  5030. ' TProc = reference to procedure;',
  5031. ' TFunc = reference to function(x: word): word;',
  5032. 'procedure Run;',
  5033. 'asm',
  5034. 'end;',
  5035. 'procedure Walk(p: TProc; f: TFunc);',
  5036. 'begin',
  5037. ' Walk(procedure asm end, function(b:word): word asm return 1+b; end);',
  5038. 'end;',
  5039. 'begin',
  5040. ' Walk(procedure',
  5041. ' asm',
  5042. ' console.log("a");',
  5043. ' end,',
  5044. ' function(x: word): word asm',
  5045. ' console.log("c");',
  5046. ' end);',
  5047. '']);
  5048. ConvertProgram;
  5049. CheckSource('TestAnonymousProc_AsmDelphi',
  5050. LinesToStr([ // statements
  5051. 'this.Run = function () {',
  5052. '};',
  5053. 'this.Walk = function (p, f) {',
  5054. ' $mod.Walk(function () {',
  5055. ' }, function (b) {',
  5056. ' return 1+b;',
  5057. ' });',
  5058. '};',
  5059. '']),
  5060. LinesToStr([
  5061. '$mod.Walk(function () {',
  5062. ' console.log("a");',
  5063. '}, function (x) {',
  5064. ' console.log("c");',
  5065. '});',
  5066. '']));
  5067. end;
  5068. procedure TTestModule.TestEnum_Name;
  5069. begin
  5070. StartProgram(false);
  5071. Add('type TMyEnum = (Red, Green, Blue);');
  5072. Add('var e: TMyEnum;');
  5073. Add('var f: TMyEnum = Blue;');
  5074. Add('begin');
  5075. Add(' e:=green;');
  5076. Add(' e:=default(TMyEnum);');
  5077. ConvertProgram;
  5078. CheckSource('TestEnum_Name',
  5079. LinesToStr([ // statements
  5080. 'this.TMyEnum = {',
  5081. ' "0":"Red",',
  5082. ' Red:0,',
  5083. ' "1":"Green",',
  5084. ' Green:1,',
  5085. ' "2":"Blue",',
  5086. ' Blue:2',
  5087. ' };',
  5088. 'this.e = 0;',
  5089. 'this.f = this.TMyEnum.Blue;'
  5090. ]),
  5091. LinesToStr([
  5092. '$mod.e=$mod.TMyEnum.Green;',
  5093. '$mod.e=$mod.TMyEnum.Red;'
  5094. ]));
  5095. end;
  5096. procedure TTestModule.TestEnum_Number;
  5097. begin
  5098. Converter.Options:=Converter.Options+[coEnumNumbers];
  5099. StartProgram(false);
  5100. Add('type TMyEnum = (Red, Green);');
  5101. Add('var');
  5102. Add(' e: TMyEnum;');
  5103. Add(' f: TMyEnum = Green;');
  5104. Add(' i: longint;');
  5105. Add('begin');
  5106. Add(' e:=green;');
  5107. Add(' i:=longint(e);');
  5108. ConvertProgram;
  5109. CheckSource('TestEnumNumber',
  5110. LinesToStr([ // statements
  5111. 'this.TMyEnum = {',
  5112. ' "0":"Red",',
  5113. ' Red:0,',
  5114. ' "1":"Green",',
  5115. ' Green:1',
  5116. ' };',
  5117. 'this.e = 0;',
  5118. 'this.f = 1;',
  5119. 'this.i = 0;'
  5120. ]),
  5121. LinesToStr([
  5122. '$mod.e=1;',
  5123. '$mod.i=$mod.e;'
  5124. ]));
  5125. end;
  5126. procedure TTestModule.TestEnum_ConstFail;
  5127. begin
  5128. StartProgram(false);
  5129. Add([
  5130. 'type TMyEnum = (Red = 100, Green = 101);',
  5131. 'var',
  5132. ' e: TMyEnum;',
  5133. ' f: TMyEnum = Green;',
  5134. 'begin',
  5135. ' e:=green;']);
  5136. SetExpectedPasResolverError('not yet implemented: Red:TPasEnumValue [20180126202434] "enum const"',3002);
  5137. ConvertProgram;
  5138. end;
  5139. procedure TTestModule.TestEnum_Functions;
  5140. begin
  5141. StartProgram(false);
  5142. Add([
  5143. 'type TMyEnum = (Red, Green);',
  5144. 'procedure DoIt(var e: TMyEnum; var i: word);',
  5145. 'var',
  5146. ' v: longint;',
  5147. ' s: string;',
  5148. 'begin',
  5149. ' val(s,e,v);',
  5150. ' val(s,e,i);',
  5151. 'end;',
  5152. 'var',
  5153. ' e: TMyEnum;',
  5154. ' i: longint;',
  5155. ' s: string;',
  5156. ' b: boolean;',
  5157. 'begin',
  5158. ' i:=ord(red);',
  5159. ' i:=ord(green);',
  5160. ' i:=ord(e);',
  5161. ' i:=ord(b);',
  5162. ' e:=low(tmyenum);',
  5163. ' e:=low(e);',
  5164. ' b:=low(boolean);',
  5165. ' e:=high(tmyenum);',
  5166. ' e:=high(e);',
  5167. ' b:=high(boolean);',
  5168. ' e:=pred(green);',
  5169. ' e:=pred(e);',
  5170. ' b:=pred(b);',
  5171. ' e:=succ(red);',
  5172. ' e:=succ(e);',
  5173. ' b:=succ(b);',
  5174. ' e:=tmyenum(1);',
  5175. ' e:=tmyenum(i);',
  5176. ' s:=str(e);',
  5177. ' str(e,s);',
  5178. ' str(red,s);',
  5179. ' s:=str(e:3);',
  5180. ' writestr(s,e:3,red);',
  5181. ' val(s,e,i);',
  5182. ' i:=longint(e);']);
  5183. ConvertProgram;
  5184. CheckSource('TestEnum_Functions',
  5185. LinesToStr([ // statements
  5186. 'this.TMyEnum = {',
  5187. ' "0":"Red",',
  5188. ' Red:0,',
  5189. ' "1":"Green",',
  5190. ' Green:1',
  5191. ' };',
  5192. 'this.DoIt = function (e, i) {',
  5193. ' var v = 0;',
  5194. ' var s = "";',
  5195. ' e.set(rtl.valEnum(s, $mod.TMyEnum, function (w) {',
  5196. ' v = w;',
  5197. ' }));',
  5198. ' e.set(rtl.valEnum(s, $mod.TMyEnum, i.set));',
  5199. '};',
  5200. 'this.e = 0;',
  5201. 'this.i = 0;',
  5202. 'this.s = "";',
  5203. 'this.b = false;',
  5204. '']),
  5205. LinesToStr([
  5206. '$mod.i=$mod.TMyEnum.Red;',
  5207. '$mod.i=$mod.TMyEnum.Green;',
  5208. '$mod.i=$mod.e;',
  5209. '$mod.i=$mod.b+0;',
  5210. '$mod.e=$mod.TMyEnum.Red;',
  5211. '$mod.e=$mod.TMyEnum.Red;',
  5212. '$mod.b=false;',
  5213. '$mod.e=$mod.TMyEnum.Green;',
  5214. '$mod.e=$mod.TMyEnum.Green;',
  5215. '$mod.b=true;',
  5216. '$mod.e=$mod.TMyEnum.Green-1;',
  5217. '$mod.e=$mod.e-1;',
  5218. '$mod.b=false;',
  5219. '$mod.e=$mod.TMyEnum.Red+1;',
  5220. '$mod.e=$mod.e+1;',
  5221. '$mod.b=true;',
  5222. '$mod.e=1;',
  5223. '$mod.e=$mod.i;',
  5224. '$mod.s = $mod.TMyEnum[$mod.e];',
  5225. '$mod.s = $mod.TMyEnum[$mod.e];',
  5226. '$mod.s = $mod.TMyEnum[$mod.TMyEnum.Red];',
  5227. '$mod.s = rtl.spaceLeft($mod.TMyEnum[$mod.e], 3);',
  5228. '$mod.s = rtl.spaceLeft($mod.TMyEnum[$mod.e], 3)+$mod.TMyEnum[$mod.TMyEnum.Red];',
  5229. '$mod.e = rtl.valEnum($mod.s, $mod.TMyEnum, function (v) {',
  5230. ' $mod.i = v;',
  5231. '});',
  5232. '$mod.i=$mod.e;',
  5233. '']));
  5234. end;
  5235. procedure TTestModule.TestEnumRg_Functions;
  5236. begin
  5237. StartProgram(false);
  5238. Add([
  5239. 'type',
  5240. ' TEnum = (Red, Green, Blue);',
  5241. ' TEnumRg = Green..Blue;',
  5242. 'procedure DoIt(var e: TEnumRg; var i: word);',
  5243. 'var',
  5244. ' v: longint;',
  5245. ' s: string;',
  5246. 'begin',
  5247. ' val(s,e,v);',
  5248. ' val(s,e,i);',
  5249. 'end;',
  5250. 'var',
  5251. ' e: TEnumRg;',
  5252. ' i: longint;',
  5253. ' s: string;',
  5254. 'begin',
  5255. ' i:=ord(green);',
  5256. ' i:=ord(e);',
  5257. ' e:=low(tenumrg);',
  5258. ' e:=low(e);',
  5259. ' e:=high(tenumrg);',
  5260. ' e:=high(e);',
  5261. ' e:=pred(blue);',
  5262. ' e:=pred(e);',
  5263. ' e:=succ(green);',
  5264. ' e:=succ(e);',
  5265. ' e:=tenumrg(1);',
  5266. ' e:=tenumrg(i);',
  5267. ' s:=str(e);',
  5268. ' str(e,s);',
  5269. ' str(red,s);',
  5270. ' s:=str(e:3);',
  5271. ' writestr(s,e:3,blue);',
  5272. ' val(s,e,i);',
  5273. ' i:=longint(e);']);
  5274. ConvertProgram;
  5275. CheckSource('TestEnumRg_Functions',
  5276. LinesToStr([ // statements
  5277. 'this.TEnum = {',
  5278. ' "0":"Red",',
  5279. ' Red:0,',
  5280. ' "1":"Green",',
  5281. ' Green:1,',
  5282. ' "2":"Blue",',
  5283. ' Blue:2',
  5284. ' };',
  5285. 'this.DoIt = function (e, i) {',
  5286. ' var v = 0;',
  5287. ' var s = "";',
  5288. ' e.set(rtl.valEnum(s, $mod.TEnum, function (w) {',
  5289. ' v = w;',
  5290. ' }));',
  5291. ' e.set(rtl.valEnum(s, $mod.TEnum, i.set));',
  5292. '};',
  5293. 'this.e = this.TEnum.Green;',
  5294. 'this.i = 0;',
  5295. 'this.s = "";',
  5296. '']),
  5297. LinesToStr([
  5298. '$mod.i=$mod.TEnum.Green;',
  5299. '$mod.i=$mod.e;',
  5300. '$mod.e=$mod.TEnum.Green;',
  5301. '$mod.e=$mod.TEnum.Green;',
  5302. '$mod.e=$mod.TEnum.Blue;',
  5303. '$mod.e=$mod.TEnum.Blue;',
  5304. '$mod.e=$mod.TEnum.Blue-1;',
  5305. '$mod.e=$mod.e-1;',
  5306. '$mod.e=$mod.TEnum.Green+1;',
  5307. '$mod.e=$mod.e+1;',
  5308. '$mod.e=1;',
  5309. '$mod.e=$mod.i;',
  5310. '$mod.s = $mod.TEnum[$mod.e];',
  5311. '$mod.s = $mod.TEnum[$mod.e];',
  5312. '$mod.s = $mod.TEnum[$mod.TEnum.Red];',
  5313. '$mod.s = rtl.spaceLeft($mod.TEnum[$mod.e], 3);',
  5314. '$mod.s = rtl.spaceLeft($mod.TEnum[$mod.e], 3)+$mod.TEnum[$mod.TEnum.Blue];',
  5315. '$mod.e = rtl.valEnum($mod.s, $mod.TEnum, function (v) {',
  5316. ' $mod.i = v;',
  5317. '});',
  5318. '$mod.i=$mod.e;',
  5319. '']));
  5320. end;
  5321. procedure TTestModule.TestEnum_AsParams;
  5322. begin
  5323. StartProgram(false);
  5324. Add('type TEnum = (Red,Blue);');
  5325. Add('procedure DoIt(vG: TEnum; const vH: TEnum; var vI: TEnum);');
  5326. Add('var vJ: TEnum;');
  5327. Add('begin');
  5328. Add(' vg:=vg;');
  5329. Add(' vj:=vh;');
  5330. Add(' vi:=vi;');
  5331. Add(' doit(vg,vg,vg);');
  5332. Add(' doit(vh,vh,vj);');
  5333. Add(' doit(vi,vi,vi);');
  5334. Add(' doit(vj,vj,vj);');
  5335. Add('end;');
  5336. Add('var i: TEnum;');
  5337. Add('begin');
  5338. Add(' doit(i,i,i);');
  5339. ConvertProgram;
  5340. CheckSource('TestEnum_AsParams',
  5341. LinesToStr([ // statements
  5342. 'this.TEnum = {',
  5343. ' "0": "Red",',
  5344. ' Red: 0,',
  5345. ' "1": "Blue",',
  5346. ' Blue: 1',
  5347. '};',
  5348. 'this.DoIt = function (vG,vH,vI) {',
  5349. ' var vJ = 0;',
  5350. ' vG = vG;',
  5351. ' vJ = vH;',
  5352. ' vI.set(vI.get());',
  5353. ' $mod.DoIt(vG, vG, {',
  5354. ' get: function () {',
  5355. ' return vG;',
  5356. ' },',
  5357. ' set: function (v) {',
  5358. ' vG = v;',
  5359. ' }',
  5360. ' });',
  5361. ' $mod.DoIt(vH, vH, {',
  5362. ' get: function () {',
  5363. ' return vJ;',
  5364. ' },',
  5365. ' set: function (v) {',
  5366. ' vJ = v;',
  5367. ' }',
  5368. ' });',
  5369. ' $mod.DoIt(vI.get(), vI.get(), vI);',
  5370. ' $mod.DoIt(vJ, vJ, {',
  5371. ' get: function () {',
  5372. ' return vJ;',
  5373. ' },',
  5374. ' set: function (v) {',
  5375. ' vJ = v;',
  5376. ' }',
  5377. ' });',
  5378. '};',
  5379. 'this.i = 0;'
  5380. ]),
  5381. LinesToStr([
  5382. '$mod.DoIt($mod.i,$mod.i,{',
  5383. ' p: $mod,',
  5384. ' get: function () {',
  5385. ' return this.p.i;',
  5386. ' },',
  5387. ' set: function (v) {',
  5388. ' this.p.i = v;',
  5389. ' }',
  5390. '});'
  5391. ]));
  5392. end;
  5393. procedure TTestModule.TestEnumRange_Array;
  5394. begin
  5395. StartProgram(false);
  5396. Add([
  5397. 'type',
  5398. ' TEnum = (Red, Green, Blue);',
  5399. ' TEnumRg = green..blue;',
  5400. ' TArr = array[TEnumRg] of byte;',
  5401. ' TArr2 = array[green..blue] of byte;',
  5402. 'var',
  5403. ' a: TArr;',
  5404. ' b: TArr = (3,4);',
  5405. ' c: TArr2 = (5,6);',
  5406. 'begin',
  5407. ' a[green] := b[blue];',
  5408. ' c[green] := c[blue];',
  5409. '']);
  5410. ConvertProgram;
  5411. CheckSource('TestEnumRange_Array',
  5412. LinesToStr([ // statements
  5413. 'this.TEnum = {',
  5414. ' "0": "Red",',
  5415. ' Red: 0,',
  5416. ' "1": "Green",',
  5417. ' Green: 1,',
  5418. ' "2": "Blue",',
  5419. ' Blue: 2',
  5420. '};',
  5421. 'this.a = rtl.arraySetLength(null, 0, 2);',
  5422. 'this.b = [3, 4];',
  5423. 'this.c = [5, 6];',
  5424. '']),
  5425. LinesToStr([
  5426. ' $mod.a[$mod.TEnum.Green - 1] = $mod.b[$mod.TEnum.Blue - 1];',
  5427. ' $mod.c[$mod.TEnum.Green - 1] = $mod.c[$mod.TEnum.Blue - 1];',
  5428. '']));
  5429. end;
  5430. procedure TTestModule.TestEnum_ForIn;
  5431. begin
  5432. StartProgram(false);
  5433. Add([
  5434. 'type',
  5435. ' TEnum = (Red, Green, Blue);',
  5436. ' TEnumRg = green..blue;',
  5437. ' TArr = array[TEnum] of byte;',
  5438. ' TArrRg = array[TEnumRg] of byte;',
  5439. 'var',
  5440. ' e: TEnum;',
  5441. ' a1: TArr = (3,4,5);',
  5442. ' a2: TArrRg = (11,12);',
  5443. ' b: byte;',
  5444. 'begin',
  5445. ' for e in TEnum do ;',
  5446. ' for e in TEnumRg do ;',
  5447. ' for e in TArr do ;',
  5448. ' for e in TArrRg do ;',
  5449. ' for b in a1 do ;',
  5450. ' for b in a2 do ;',
  5451. '']);
  5452. ConvertProgram;
  5453. CheckSource('TestEnum_ForIn',
  5454. LinesToStr([ // statements
  5455. 'this.TEnum = {',
  5456. ' "0": "Red",',
  5457. ' Red: 0,',
  5458. ' "1": "Green",',
  5459. ' Green: 1,',
  5460. ' "2": "Blue",',
  5461. ' Blue: 2',
  5462. '};',
  5463. 'this.e = 0;',
  5464. 'this.a1 = [3, 4, 5];',
  5465. 'this.a2 = [11, 12];',
  5466. 'this.b = 0;',
  5467. '']),
  5468. LinesToStr([
  5469. ' for ($mod.e = 0; $mod.e <= 2; $mod.e++) ;',
  5470. ' for ($mod.e = 1; $mod.e <= 2; $mod.e++) ;',
  5471. ' for ($mod.e = 0; $mod.e <= 2; $mod.e++) ;',
  5472. ' for ($mod.e = 1; $mod.e <= 2; $mod.e++) ;',
  5473. ' for (var $in = $mod.a1, $l = 0, $end = rtl.length($in) - 1; $l <= $end; $l++) $mod.b = $in[$l];',
  5474. ' for (var $in1 = $mod.a2, $l1 = 0, $end1 = rtl.length($in1) - 1; $l1 <= $end1; $l1++) $mod.b = $in1[$l1];',
  5475. '']));
  5476. end;
  5477. procedure TTestModule.TestEnum_ScopedNumber;
  5478. begin
  5479. Converter.Options:=Converter.Options+[coEnumNumbers];
  5480. StartProgram(false);
  5481. Add([
  5482. 'type',
  5483. ' TEnum = (Red, Green);',
  5484. 'var',
  5485. ' e: TEnum;',
  5486. 'begin',
  5487. ' e:=TEnum.Green;',
  5488. '']);
  5489. ConvertProgram;
  5490. CheckSource('TestEnum_ScopedNumber',
  5491. LinesToStr([ // statements
  5492. 'this.TEnum = {',
  5493. ' "0": "Red",',
  5494. ' Red: 0,',
  5495. ' "1": "Green",',
  5496. ' Green: 1',
  5497. '};',
  5498. 'this.e = 0;',
  5499. '']),
  5500. LinesToStr([
  5501. '$mod.e = 1;']));
  5502. end;
  5503. procedure TTestModule.TestEnum_InFunction;
  5504. begin
  5505. StartProgram(false);
  5506. Add([
  5507. 'const TEnum = 3;',
  5508. 'procedure DoIt;',
  5509. 'type',
  5510. ' TEnum = (Red, Green, Blue);',
  5511. ' procedure Sub;',
  5512. ' type',
  5513. ' TEnumSub = (Left, Right);',
  5514. ' var',
  5515. ' es: TEnumSub;',
  5516. ' begin',
  5517. ' es:=Left;',
  5518. ' end;',
  5519. 'var',
  5520. ' e, e2: TEnum;',
  5521. 'begin',
  5522. ' if e in [red,blue] then e2:=e;',
  5523. 'end;',
  5524. 'begin']);
  5525. ConvertProgram;
  5526. CheckSource('TestEnum_InFunction',
  5527. LinesToStr([ // statements
  5528. 'this.TEnum = 3;',
  5529. 'var TEnum$1 = {',
  5530. ' "0":"Red",',
  5531. ' Red:0,',
  5532. ' "1":"Green",',
  5533. ' Green:1,',
  5534. ' "2":"Blue",',
  5535. ' Blue:2',
  5536. ' };',
  5537. 'var TEnumSub = {',
  5538. ' "0": "Left",',
  5539. ' Left: 0,',
  5540. ' "1": "Right",',
  5541. ' Right: 1',
  5542. '};',
  5543. 'this.DoIt = function () {',
  5544. ' function Sub() {',
  5545. ' var es = 0;',
  5546. ' es = TEnumSub.Left;',
  5547. ' };',
  5548. ' var e = 0;',
  5549. ' var e2 = 0;',
  5550. ' if (e in rtl.createSet(TEnum$1.Red, TEnum$1.Blue)) e2 = e;',
  5551. '};',
  5552. '']),
  5553. LinesToStr([
  5554. '']));
  5555. end;
  5556. procedure TTestModule.TestSet_Enum;
  5557. begin
  5558. StartProgram(false);
  5559. Add([
  5560. 'type',
  5561. ' TColor = (Red, Green, Blue);',
  5562. ' TColors = set of TColor;',
  5563. 'var',
  5564. ' c: TColor;',
  5565. ' s: TColors;',
  5566. ' t: TColors = [];',
  5567. ' u: TColors = [Red];',
  5568. 'begin',
  5569. ' s:=[];',
  5570. ' s:=[Green];',
  5571. ' s:=[Green,Blue];',
  5572. ' s:=[Red..Blue];',
  5573. ' s:=[Red,Green..Blue];',
  5574. ' s:=[Red,c];',
  5575. ' s:=t;',
  5576. ' s:=default(TColors);',
  5577. '']);
  5578. ConvertProgram;
  5579. CheckSource('TestSet',
  5580. LinesToStr([ // statements
  5581. 'this.TColor = {',
  5582. ' "0":"Red",',
  5583. ' Red:0,',
  5584. ' "1":"Green",',
  5585. ' Green:1,',
  5586. ' "2":"Blue",',
  5587. ' Blue:2',
  5588. ' };',
  5589. 'this.c = 0;',
  5590. 'this.s = {};',
  5591. 'this.t = {};',
  5592. 'this.u = rtl.createSet(this.TColor.Red);'
  5593. ]),
  5594. LinesToStr([
  5595. '$mod.s={};',
  5596. '$mod.s=rtl.createSet($mod.TColor.Green);',
  5597. '$mod.s=rtl.createSet($mod.TColor.Green,$mod.TColor.Blue);',
  5598. '$mod.s=rtl.createSet(null,$mod.TColor.Red,$mod.TColor.Blue);',
  5599. '$mod.s=rtl.createSet($mod.TColor.Red,null,$mod.TColor.Green,$mod.TColor.Blue);',
  5600. '$mod.s=rtl.createSet($mod.TColor.Red,$mod.c);',
  5601. '$mod.s=rtl.refSet($mod.t);',
  5602. '$mod.s={};',
  5603. '']));
  5604. end;
  5605. procedure TTestModule.TestSet_Operators;
  5606. begin
  5607. StartProgram(false);
  5608. Add('type');
  5609. Add(' TColor = (Red, Green, Blue);');
  5610. Add(' TColors = set of tcolor;');
  5611. Add('var');
  5612. Add(' vC: TColor;');
  5613. Add(' vS: TColors;');
  5614. Add(' vT: TColors;');
  5615. Add(' vU: TColors;');
  5616. Add(' B: boolean;');
  5617. Add('begin');
  5618. Add(' include(vs,green);');
  5619. Add(' exclude(vs,vc);');
  5620. Add(' vs:=vt+vu;');
  5621. Add(' vs:=vt+[red];');
  5622. Add(' vs:=[red]+vt;');
  5623. Add(' vs:=[red]+[green];');
  5624. Add(' vs:=vt-vu;');
  5625. Add(' vs:=vt-[red];');
  5626. Add(' vs:=[red]-vt;');
  5627. Add(' vs:=[red]-[green];');
  5628. Add(' vs:=vt*vu;');
  5629. Add(' vs:=vt*[red];');
  5630. Add(' vs:=[red]*vt;');
  5631. Add(' vs:=[red]*[green];');
  5632. Add(' vs:=vt><vu;');
  5633. Add(' vs:=vt><[red];');
  5634. Add(' vs:=[red]><vt;');
  5635. Add(' vs:=[red]><[green];');
  5636. Add(' b:=vt=vu;');
  5637. Add(' b:=vt=[red];');
  5638. Add(' b:=[red]=vt;');
  5639. Add(' b:=[red]=[green];');
  5640. Add(' b:=vt<>vu;');
  5641. Add(' b:=vt<>[red];');
  5642. Add(' b:=[red]<>vt;');
  5643. Add(' b:=[red]<>[green];');
  5644. Add(' b:=vt<=vu;');
  5645. Add(' b:=vt<=[red];');
  5646. Add(' b:=[red]<=vt;');
  5647. Add(' b:=[red]<=[green];');
  5648. Add(' b:=vt>=vu;');
  5649. Add(' b:=vt>=[red];');
  5650. Add(' b:=[red]>=vt;');
  5651. Add(' b:=[red]>=[green];');
  5652. ConvertProgram;
  5653. CheckSource('TestSet_Operators',
  5654. LinesToStr([ // statements
  5655. 'this.TColor = {',
  5656. ' "0":"Red",',
  5657. ' Red:0,',
  5658. ' "1":"Green",',
  5659. ' Green:1,',
  5660. ' "2":"Blue",',
  5661. ' Blue:2',
  5662. ' };',
  5663. 'this.vC = 0;',
  5664. 'this.vS = {};',
  5665. 'this.vT = {};',
  5666. 'this.vU = {};',
  5667. 'this.B = false;'
  5668. ]),
  5669. LinesToStr([
  5670. '$mod.vS = rtl.includeSet($mod.vS,$mod.TColor.Green);',
  5671. '$mod.vS = rtl.excludeSet($mod.vS,$mod.vC);',
  5672. '$mod.vS = rtl.unionSet($mod.vT, $mod.vU);',
  5673. '$mod.vS = rtl.unionSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5674. '$mod.vS = rtl.unionSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5675. '$mod.vS = rtl.unionSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5676. '$mod.vS = rtl.diffSet($mod.vT, $mod.vU);',
  5677. '$mod.vS = rtl.diffSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5678. '$mod.vS = rtl.diffSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5679. '$mod.vS = rtl.diffSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5680. '$mod.vS = rtl.intersectSet($mod.vT, $mod.vU);',
  5681. '$mod.vS = rtl.intersectSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5682. '$mod.vS = rtl.intersectSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5683. '$mod.vS = rtl.intersectSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5684. '$mod.vS = rtl.symDiffSet($mod.vT, $mod.vU);',
  5685. '$mod.vS = rtl.symDiffSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5686. '$mod.vS = rtl.symDiffSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5687. '$mod.vS = rtl.symDiffSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5688. '$mod.B = rtl.eqSet($mod.vT, $mod.vU);',
  5689. '$mod.B = rtl.eqSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5690. '$mod.B = rtl.eqSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5691. '$mod.B = rtl.eqSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5692. '$mod.B = rtl.neSet($mod.vT, $mod.vU);',
  5693. '$mod.B = rtl.neSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5694. '$mod.B = rtl.neSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5695. '$mod.B = rtl.neSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5696. '$mod.B = rtl.leSet($mod.vT, $mod.vU);',
  5697. '$mod.B = rtl.leSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5698. '$mod.B = rtl.leSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5699. '$mod.B = rtl.leSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5700. '$mod.B = rtl.geSet($mod.vT, $mod.vU);',
  5701. '$mod.B = rtl.geSet($mod.vT, rtl.createSet($mod.TColor.Red));',
  5702. '$mod.B = rtl.geSet(rtl.createSet($mod.TColor.Red), $mod.vT);',
  5703. '$mod.B = rtl.geSet(rtl.createSet($mod.TColor.Red), rtl.createSet($mod.TColor.Green));',
  5704. '']));
  5705. end;
  5706. procedure TTestModule.TestSet_Operator_In;
  5707. begin
  5708. StartProgram(false);
  5709. Add([
  5710. 'type',
  5711. ' TColor = (Red, Green, Blue);',
  5712. ' TColors = set of tcolor;',
  5713. ' TColorRg = green..blue;',
  5714. 'var',
  5715. ' vC: tcolor;',
  5716. ' vT: tcolors;',
  5717. ' B: boolean;',
  5718. ' rg: TColorRg;',
  5719. 'begin',
  5720. ' b:=red in vt;',
  5721. ' b:=vc in vt;',
  5722. ' b:=green in [red..blue];',
  5723. ' b:=vc in [red..blue];',
  5724. ' ',
  5725. ' if red in vt then ;',
  5726. ' while vC in vt do ;',
  5727. ' repeat',
  5728. ' until vC in vt;',
  5729. ' if rg in [green..blue] then ;',
  5730. '']);
  5731. ConvertProgram;
  5732. CheckSource('TestSet_Operator_In',
  5733. LinesToStr([ // statements
  5734. 'this.TColor = {',
  5735. ' "0":"Red",',
  5736. ' Red:0,',
  5737. ' "1":"Green",',
  5738. ' Green:1,',
  5739. ' "2":"Blue",',
  5740. ' Blue:2',
  5741. ' };',
  5742. 'this.vC = 0;',
  5743. 'this.vT = {};',
  5744. 'this.B = false;',
  5745. 'this.rg = this.TColor.Green;',
  5746. '']),
  5747. LinesToStr([
  5748. '$mod.B = $mod.TColor.Red in $mod.vT;',
  5749. '$mod.B = $mod.vC in $mod.vT;',
  5750. '$mod.B = $mod.TColor.Green in rtl.createSet(null, $mod.TColor.Red, $mod.TColor.Blue);',
  5751. '$mod.B = $mod.vC in rtl.createSet(null, $mod.TColor.Red, $mod.TColor.Blue);',
  5752. 'if ($mod.TColor.Red in $mod.vT) ;',
  5753. 'while ($mod.vC in $mod.vT) {',
  5754. '};',
  5755. 'do {',
  5756. '} while (!($mod.vC in $mod.vT));',
  5757. 'if ($mod.rg in rtl.createSet(null, $mod.TColor.Green, $mod.TColor.Blue)) ;',
  5758. '']));
  5759. end;
  5760. procedure TTestModule.TestSet_Functions;
  5761. begin
  5762. StartProgram(false);
  5763. Add('type');
  5764. Add(' TMyEnum = (Red, Green);');
  5765. Add(' TMyEnums = set of TMyEnum;');
  5766. Add('var');
  5767. Add(' e: TMyEnum;');
  5768. Add(' s: TMyEnums;');
  5769. Add('begin');
  5770. Add(' e:=Low(TMyEnums);');
  5771. Add(' e:=Low(s);');
  5772. Add(' e:=High(TMyEnums);');
  5773. Add(' e:=High(s);');
  5774. ConvertProgram;
  5775. CheckSource('TestSetFunctions',
  5776. LinesToStr([ // statements
  5777. 'this.TMyEnum = {',
  5778. ' "0":"Red",',
  5779. ' Red:0,',
  5780. ' "1":"Green",',
  5781. ' Green:1',
  5782. ' };',
  5783. 'this.e = 0;',
  5784. 'this.s = {};'
  5785. ]),
  5786. LinesToStr([
  5787. '$mod.e=$mod.TMyEnum.Red;',
  5788. '$mod.e=$mod.TMyEnum.Red;',
  5789. '$mod.e=$mod.TMyEnum.Green;',
  5790. '$mod.e=$mod.TMyEnum.Green;',
  5791. '']));
  5792. end;
  5793. procedure TTestModule.TestSet_PassAsArgClone;
  5794. begin
  5795. StartProgram(false);
  5796. Add('type');
  5797. Add(' TMyEnum = (Red, Green);');
  5798. Add(' TMyEnums = set of TMyEnum;');
  5799. Add('procedure DoDefault(s: tmyenums); begin end;');
  5800. Add('procedure DoConst(const s: tmyenums); begin end;');
  5801. Add('var');
  5802. Add(' aSet: tmyenums;');
  5803. Add('begin');
  5804. Add(' dodefault(aset);');
  5805. Add(' doconst(aset);');
  5806. ConvertProgram;
  5807. CheckSource('TestSetFunctions',
  5808. LinesToStr([ // statements
  5809. 'this.TMyEnum = {',
  5810. ' "0":"Red",',
  5811. ' Red:0,',
  5812. ' "1":"Green",',
  5813. ' Green:1',
  5814. ' };',
  5815. 'this.DoDefault = function (s) {',
  5816. '};',
  5817. 'this.DoConst = function (s) {',
  5818. '};',
  5819. 'this.aSet = {};'
  5820. ]),
  5821. LinesToStr([
  5822. '$mod.DoDefault(rtl.refSet($mod.aSet));',
  5823. '$mod.DoConst($mod.aSet);',
  5824. '']));
  5825. end;
  5826. procedure TTestModule.TestSet_AsParams;
  5827. begin
  5828. StartProgram(false);
  5829. Add([
  5830. 'type TEnum = (Red,Blue);',
  5831. 'type TEnums = set of TEnum;',
  5832. 'function DoIt(vG: TEnums; const vH: TEnums; var vI: TEnums): TEnums;',
  5833. 'var vJ: TEnums;',
  5834. 'begin',
  5835. ' Include(vg,red);',
  5836. ' Include(result,blue);',
  5837. ' vg:=vg;',
  5838. ' vj:=vh;',
  5839. ' vi:=vi;',
  5840. ' doit(vg,vg,vg);',
  5841. ' doit(vh,vh,vj);',
  5842. ' doit(vi,vi,vi);',
  5843. ' doit(vj,vj,vj);',
  5844. 'end;',
  5845. 'var i: TEnums;',
  5846. 'begin',
  5847. ' doit(i,i,i);']);
  5848. ConvertProgram;
  5849. CheckSource('TestSet_AsParams',
  5850. LinesToStr([ // statements
  5851. 'this.TEnum = {',
  5852. ' "0": "Red",',
  5853. ' Red: 0,',
  5854. ' "1": "Blue",',
  5855. ' Blue: 1',
  5856. '};',
  5857. 'this.DoIt = function (vG,vH,vI) {',
  5858. ' var Result = {};',
  5859. ' var vJ = {};',
  5860. ' vG = rtl.includeSet(vG, $mod.TEnum.Red);',
  5861. ' Result = rtl.includeSet(Result, $mod.TEnum.Blue);',
  5862. ' vG = rtl.refSet(vG);',
  5863. ' vJ = rtl.refSet(vH);',
  5864. ' vI.set(rtl.refSet(vI.get()));',
  5865. ' $mod.DoIt(rtl.refSet(vG), vG, {',
  5866. ' get: function () {',
  5867. ' return vG;',
  5868. ' },',
  5869. ' set: function (v) {',
  5870. ' vG = v;',
  5871. ' }',
  5872. ' });',
  5873. ' $mod.DoIt(rtl.refSet(vH), vH, {',
  5874. ' get: function () {',
  5875. ' return vJ;',
  5876. ' },',
  5877. ' set: function (v) {',
  5878. ' vJ = v;',
  5879. ' }',
  5880. ' });',
  5881. ' $mod.DoIt(rtl.refSet(vI.get()), vI.get(), vI);',
  5882. ' $mod.DoIt(rtl.refSet(vJ), vJ, {',
  5883. ' get: function () {',
  5884. ' return vJ;',
  5885. ' },',
  5886. ' set: function (v) {',
  5887. ' vJ = v;',
  5888. ' }',
  5889. ' });',
  5890. ' return Result;',
  5891. '};',
  5892. 'this.i = {};'
  5893. ]),
  5894. LinesToStr([
  5895. '$mod.DoIt(rtl.refSet($mod.i),$mod.i,{',
  5896. ' p: $mod,',
  5897. ' get: function () {',
  5898. ' return this.p.i;',
  5899. ' },',
  5900. ' set: function (v) {',
  5901. ' this.p.i = v;',
  5902. ' }',
  5903. '});'
  5904. ]));
  5905. end;
  5906. procedure TTestModule.TestSet_Property;
  5907. begin
  5908. StartProgram(false);
  5909. Add('type');
  5910. Add(' TEnum = (Red,Blue);');
  5911. Add(' TEnums = set of TEnum;');
  5912. Add(' TObject = class');
  5913. Add(' function GetColors: TEnums; external name ''GetColors'';');
  5914. Add(' procedure SetColors(const Value: TEnums); external name ''SetColors'';');
  5915. Add(' property Colors: TEnums read GetColors write SetColors;');
  5916. Add(' end;');
  5917. Add('procedure DoIt(i: TEnums; const j: TEnums; var k: TEnums; out l: TEnums);');
  5918. Add('begin end;');
  5919. Add('var Obj: TObject;');
  5920. Add('begin');
  5921. Add(' Include(Obj.Colors,Red);');
  5922. Add(' Exclude(Obj.Colors,Red);');
  5923. //Add(' DoIt(Obj.Colors,Obj.Colors,Obj.Colors,Obj.Colors);');
  5924. ConvertProgram;
  5925. CheckSource('TestSet_Property',
  5926. LinesToStr([ // statements
  5927. 'this.TEnum = {',
  5928. ' "0": "Red",',
  5929. ' Red: 0,',
  5930. ' "1": "Blue",',
  5931. ' Blue: 1',
  5932. '};',
  5933. 'rtl.createClass(this, "TObject", null, function () {',
  5934. ' this.$init = function () {',
  5935. ' };',
  5936. ' this.$final = function () {',
  5937. ' };',
  5938. '});',
  5939. 'this.DoIt = function (i, j, k, l) {',
  5940. '};',
  5941. 'this.Obj = null;',
  5942. '']),
  5943. LinesToStr([
  5944. '$mod.Obj.SetColors(rtl.includeSet($mod.Obj.GetColors(), $mod.TEnum.Red));',
  5945. '$mod.Obj.SetColors(rtl.excludeSet($mod.Obj.GetColors(), $mod.TEnum.Red));',
  5946. '']));
  5947. end;
  5948. procedure TTestModule.TestSet_EnumConst;
  5949. begin
  5950. StartProgram(false);
  5951. Add([
  5952. 'type',
  5953. ' TEnum = (Red,Blue);',
  5954. ' TEnums = set of TEnum;',
  5955. 'const',
  5956. ' Orange = red;',
  5957. 'var',
  5958. ' Enum: tenum;',
  5959. ' Enums: tenums;',
  5960. 'begin',
  5961. ' Include(enums,orange);',
  5962. ' Exclude(enums,orange);',
  5963. ' if orange in enums then;',
  5964. ' if orange in [orange,red] then;']);
  5965. ConvertProgram;
  5966. CheckSource('TestSet_EnumConst',
  5967. LinesToStr([ // statements
  5968. 'this.TEnum = {',
  5969. ' "0": "Red",',
  5970. ' Red: 0,',
  5971. ' "1": "Blue",',
  5972. ' Blue: 1',
  5973. '};',
  5974. 'this.Orange = this.TEnum.Red;',
  5975. 'this.Enum = 0;',
  5976. 'this.Enums = {};',
  5977. '']),
  5978. LinesToStr([
  5979. '$mod.Enums = rtl.includeSet($mod.Enums, $mod.TEnum.Red);',
  5980. '$mod.Enums = rtl.excludeSet($mod.Enums, $mod.TEnum.Red);',
  5981. 'if ($mod.TEnum.Red in $mod.Enums) ;',
  5982. 'if ($mod.TEnum.Red in rtl.createSet($mod.TEnum.Red, $mod.TEnum.Red)) ;',
  5983. '']));
  5984. end;
  5985. procedure TTestModule.TestSet_IntConst;
  5986. begin
  5987. StartProgram(false);
  5988. Add([
  5989. 'type',
  5990. ' TEnums = set of Byte;',
  5991. 'const',
  5992. ' Orange = 0;',
  5993. 'var',
  5994. ' Enum: byte;',
  5995. ' Enums: tenums;',
  5996. 'begin',
  5997. ' Enums:=[];',
  5998. ' Enums:=[0];',
  5999. ' Enums:=[1..2];',
  6000. //' Include(enums,orange);',
  6001. //' Exclude(enums,orange);',
  6002. ' if orange in enums then;',
  6003. ' if orange in [orange,1] then;']);
  6004. ConvertProgram;
  6005. CheckSource('TestSet_IntConst',
  6006. LinesToStr([ // statements
  6007. 'this.Orange = 0;',
  6008. 'this.Enum = 0;',
  6009. 'this.Enums = {};',
  6010. '']),
  6011. LinesToStr([
  6012. '$mod.Enums = {};',
  6013. '$mod.Enums = rtl.createSet(0);',
  6014. '$mod.Enums = rtl.createSet(null, 1, 2);',
  6015. 'if (0 in $mod.Enums) ;',
  6016. 'if (0 in rtl.createSet(0, 1)) ;',
  6017. '']));
  6018. end;
  6019. procedure TTestModule.TestSet_AnonymousEnumType;
  6020. begin
  6021. StartProgram(false);
  6022. Add('type');
  6023. Add(' TFlags = set of (red, green);');
  6024. Add('const');
  6025. Add(' favorite = red;');
  6026. Add('var');
  6027. Add(' f: TFlags;');
  6028. Add(' i: longint;');
  6029. Add('begin');
  6030. Add(' Include(f,red);');
  6031. Add(' Include(f,favorite);');
  6032. Add(' i:=ord(red);');
  6033. Add(' i:=ord(favorite);');
  6034. Add(' i:=ord(low(TFlags));');
  6035. Add(' i:=ord(low(f));');
  6036. Add(' i:=ord(low(favorite));');
  6037. Add(' i:=ord(high(TFlags));');
  6038. Add(' i:=ord(high(f));');
  6039. Add(' i:=ord(high(favorite));');
  6040. Add(' f:=[green,favorite];');
  6041. ConvertProgram;
  6042. CheckSource('TestSet_AnonymousEnumType',
  6043. LinesToStr([ // statements
  6044. 'this.TFlags$a = {',
  6045. ' "0": "red",',
  6046. ' red: 0,',
  6047. ' "1": "green",',
  6048. ' green: 1',
  6049. '};',
  6050. 'this.favorite = this.TFlags$a.red;',
  6051. 'this.f = {};',
  6052. 'this.i = 0;',
  6053. '']),
  6054. LinesToStr([
  6055. '$mod.f = rtl.includeSet($mod.f, $mod.TFlags$a.red);',
  6056. '$mod.f = rtl.includeSet($mod.f, $mod.TFlags$a.red);',
  6057. '$mod.i = $mod.TFlags$a.red;',
  6058. '$mod.i = $mod.TFlags$a.red;',
  6059. '$mod.i = $mod.TFlags$a.red;',
  6060. '$mod.i = $mod.TFlags$a.red;',
  6061. '$mod.i = $mod.TFlags$a.red;',
  6062. '$mod.i = $mod.TFlags$a.green;',
  6063. '$mod.i = $mod.TFlags$a.green;',
  6064. '$mod.i = $mod.TFlags$a.green;',
  6065. '$mod.f = rtl.createSet($mod.TFlags$a.green, $mod.TFlags$a.red);',
  6066. '']));
  6067. end;
  6068. procedure TTestModule.TestSet_AnonymousEnumTypeChar;
  6069. begin
  6070. exit;
  6071. StartProgram(false);
  6072. Add([
  6073. 'type',
  6074. ' TAtoZ = ''A''..''Z'';',
  6075. ' TSetOfAZ = set of TAtoZ;',
  6076. 'var',
  6077. ' c: char;',
  6078. ' a: TAtoZ;',
  6079. ' s: TSetOfAZ = [''P'',''A''];',
  6080. ' i: longint;',
  6081. 'begin',
  6082. ' Include(s,''S'');',
  6083. ' Include(s,c);',
  6084. ' Include(s,a);',
  6085. ' c:=low(TAtoZ);',
  6086. ' i:=ord(low(TAtoZ));',
  6087. ' a:=high(TAtoZ);',
  6088. ' a:=high(TSetOfAtoZ);',
  6089. ' s:=[a,c,''M''];',
  6090. '']);
  6091. ConvertProgram;
  6092. CheckSource('TestSet_AnonymousEnumTypeChar',
  6093. LinesToStr([ // statements
  6094. '']),
  6095. LinesToStr([
  6096. '']));
  6097. end;
  6098. procedure TTestModule.TestSet_ConstEnum;
  6099. begin
  6100. StartProgram(false);
  6101. Add([
  6102. 'type',
  6103. ' TEnum = (red,blue,green);',
  6104. ' TEnums = set of TEnum;',
  6105. 'const',
  6106. ' teAny = [low(TEnum)..high(TEnum)];',
  6107. ' teRedBlue = [low(TEnum)..pred(high(TEnum))];',
  6108. 'var',
  6109. ' e: TEnum;',
  6110. ' s: TEnums;',
  6111. 'begin',
  6112. ' if blue in teAny then;',
  6113. ' if blue in teAny+[e] then;',
  6114. ' if blue in teAny+teRedBlue then;',
  6115. ' if e in [red,blue] then;',
  6116. ' s:=teAny;',
  6117. ' s:=teAny+[e];',
  6118. ' s:=[e]+teAny;',
  6119. ' s:=teAny+teRedBlue;',
  6120. ' s:=teAny+teRedBlue+[e];',
  6121. '']);
  6122. ConvertProgram;
  6123. CheckSource('TestSet_ConstEnum',
  6124. LinesToStr([ // statements
  6125. 'this.TEnum = {',
  6126. ' "0": "red",',
  6127. ' red: 0,',
  6128. ' "1": "blue",',
  6129. ' blue: 1,',
  6130. ' "2": "green",',
  6131. ' green: 2',
  6132. '};',
  6133. 'this.teAny = rtl.createSet(null, this.TEnum.red, this.TEnum.green);',
  6134. 'this.teRedBlue = rtl.createSet(null, this.TEnum.red, this.TEnum.green - 1);',
  6135. 'this.e = 0;',
  6136. 'this.s = {};',
  6137. '']),
  6138. LinesToStr([
  6139. 'if ($mod.TEnum.blue in $mod.teAny) ;',
  6140. 'if ($mod.TEnum.blue in rtl.unionSet($mod.teAny, rtl.createSet($mod.e))) ;',
  6141. 'if ($mod.TEnum.blue in rtl.unionSet($mod.teAny, $mod.teRedBlue)) ;',
  6142. 'if ($mod.e in rtl.createSet($mod.TEnum.red, $mod.TEnum.blue)) ;',
  6143. '$mod.s = rtl.refSet($mod.teAny);',
  6144. '$mod.s = rtl.unionSet($mod.teAny, rtl.createSet($mod.e));',
  6145. '$mod.s = rtl.unionSet(rtl.createSet($mod.e), $mod.teAny);',
  6146. '$mod.s = rtl.unionSet($mod.teAny, $mod.teRedBlue);',
  6147. '$mod.s = rtl.unionSet(rtl.unionSet($mod.teAny, $mod.teRedBlue), rtl.createSet($mod.e));',
  6148. '']));
  6149. end;
  6150. procedure TTestModule.TestSet_ConstChar;
  6151. begin
  6152. StartProgram(false);
  6153. Add([
  6154. 'const',
  6155. ' LowChars = [''a''..''z''];',
  6156. ' Chars = LowChars+[''A''..''Z''];',
  6157. ' sc = [''А'', ''Я''];',
  6158. 'var',
  6159. ' c: char;',
  6160. ' s: string;',
  6161. 'begin',
  6162. ' if c in lowchars then ;',
  6163. ' if ''a'' in lowchars then ;',
  6164. ' if s[1] in lowchars then ;',
  6165. ' if c in chars then ;',
  6166. ' if c in [''a''..''z'',''_''] then ;',
  6167. ' if ''b'' in [''a''..''z'',''_''] then ;',
  6168. ' if ''Я'' in sc then ;',
  6169. ' if 3=ord('' '') then ;',
  6170. '']);
  6171. ConvertProgram;
  6172. CheckSource('TestSet_ConstChar',
  6173. LinesToStr([ // statements
  6174. 'this.LowChars = rtl.createSet(null, 97, 122);',
  6175. 'this.Chars = rtl.unionSet(this.LowChars, rtl.createSet(null, 65, 90));',
  6176. 'this.sc = rtl.createSet(1040, 1071);',
  6177. 'this.c = "";',
  6178. 'this.s = "";',
  6179. '']),
  6180. LinesToStr([
  6181. 'if ($mod.c.charCodeAt() in $mod.LowChars) ;',
  6182. 'if (97 in $mod.LowChars) ;',
  6183. 'if ($mod.s.charCodeAt(0) in $mod.LowChars) ;',
  6184. 'if ($mod.c.charCodeAt() in $mod.Chars) ;',
  6185. 'if ($mod.c.charCodeAt() in rtl.createSet(null, 97, 122, 95)) ;',
  6186. 'if (98 in rtl.createSet(null, 97, 122, 95)) ;',
  6187. 'if (1071 in $mod.sc) ;',
  6188. 'if (3 === 32) ;',
  6189. '']));
  6190. end;
  6191. procedure TTestModule.TestSet_ConstInt;
  6192. begin
  6193. StartProgram(false);
  6194. Add([
  6195. 'const',
  6196. ' Months = [1..12];',
  6197. ' Mirror = [-12..-1]+Months;',
  6198. 'var',
  6199. ' i: smallint;',
  6200. 'begin',
  6201. ' if 3 in Months then;',
  6202. ' if i in Months+[i] then;',
  6203. ' if i in Months+Mirror then;',
  6204. ' if i in [4..6,8] then;',
  6205. '']);
  6206. ConvertProgram;
  6207. CheckSource('TestSet_ConstInt',
  6208. LinesToStr([ // statements
  6209. 'this.Months = rtl.createSet(null, 1, 12);',
  6210. 'this.Mirror = rtl.unionSet(rtl.createSet(null, -12, -1), this.Months);',
  6211. 'this.i = 0;',
  6212. '']),
  6213. LinesToStr([
  6214. 'if (3 in $mod.Months) ;',
  6215. 'if ($mod.i in rtl.unionSet($mod.Months, rtl.createSet($mod.i))) ;',
  6216. 'if ($mod.i in rtl.unionSet($mod.Months, $mod.Mirror)) ;',
  6217. 'if ($mod.i in rtl.createSet(null, 4, 6, 8)) ;',
  6218. '']));
  6219. end;
  6220. procedure TTestModule.TestSet_InFunction;
  6221. begin
  6222. StartProgram(false);
  6223. Add([
  6224. 'const',
  6225. ' TEnum = 3;',
  6226. ' TSetOfEnum = 4;',
  6227. ' TSetOfAno = 5;',
  6228. 'procedure DoIt;',
  6229. 'type',
  6230. ' TEnum = (red, blue);',
  6231. ' TSetOfEnum = set of TEnum;',
  6232. ' TSetOfAno = set of (up,down);',
  6233. 'var',
  6234. ' e: TEnum;',
  6235. ' se: TSetOfEnum;',
  6236. ' sa: TSetOfAno;',
  6237. 'begin',
  6238. ' se:=[e];',
  6239. ' sa:=[up];',
  6240. 'end;',
  6241. 'begin',
  6242. '']);
  6243. ConvertProgram;
  6244. CheckSource('TestSet_InFunction',
  6245. LinesToStr([ // statements
  6246. 'this.TEnum = 3;',
  6247. 'this.TSetOfEnum = 4;',
  6248. 'this.TSetOfAno = 5;',
  6249. 'var TEnum$1 = {',
  6250. ' "0": "red",',
  6251. ' red: 0,',
  6252. ' "1": "blue",',
  6253. ' blue: 1',
  6254. '};',
  6255. 'var TSetOfAno$a = {',
  6256. ' "0": "up",',
  6257. ' up: 0,',
  6258. ' "1": "down",',
  6259. ' down: 1',
  6260. '};',
  6261. 'this.DoIt = function () {',
  6262. ' var e = 0;',
  6263. ' var se = {};',
  6264. ' var sa = {};',
  6265. ' se = rtl.createSet(e);',
  6266. ' sa = rtl.createSet(TSetOfAno$a.up);',
  6267. '};',
  6268. '']),
  6269. LinesToStr([
  6270. '']));
  6271. end;
  6272. procedure TTestModule.TestSet_ForIn;
  6273. begin
  6274. StartProgram(false);
  6275. Add([
  6276. 'type',
  6277. ' TEnum = (Red, Green, Blue);',
  6278. ' TEnumRg = green..blue;',
  6279. ' TSetOfEnum = set of TEnum;',
  6280. ' TSetOfEnumRg = set of TEnumRg;',
  6281. 'var',
  6282. ' e, e2: TEnum;',
  6283. ' er: TEnum;',
  6284. ' s: TSetOfEnum;',
  6285. 'begin',
  6286. ' for e in TSetOfEnum do ;',
  6287. ' for e in TSetOfEnumRg do ;',
  6288. ' for e in [] do e2:=e;',
  6289. ' for e in [red..green] do e2:=e;',
  6290. ' for e in [green,blue] do e2:=e;',
  6291. ' for e in [red,blue] do e2:=e;',
  6292. ' for e in s do e2:=e;',
  6293. ' for er in TSetOfEnumRg do ;',
  6294. '']);
  6295. ConvertProgram;
  6296. CheckSource('TestSet_ForIn',
  6297. LinesToStr([ // statements
  6298. 'this.TEnum = {',
  6299. ' "0":"Red",',
  6300. ' Red:0,',
  6301. ' "1":"Green",',
  6302. ' Green:1,',
  6303. ' "2":"Blue",',
  6304. ' Blue:2',
  6305. ' };',
  6306. 'this.e = 0;',
  6307. 'this.e2 = 0;',
  6308. 'this.er = 0;',
  6309. 'this.s = {};',
  6310. '']),
  6311. LinesToStr([
  6312. 'for ($mod.e = 0; $mod.e <= 2; $mod.e++) ;',
  6313. 'for ($mod.e = 1; $mod.e <= 2; $mod.e++) ;',
  6314. 'for ($mod.e = 0; $mod.e <= 1; $mod.e++) $mod.e2 = $mod.e;',
  6315. 'for ($mod.e = 1; $mod.e <= 2; $mod.e++) $mod.e2 = $mod.e;',
  6316. 'for ($mod.e in rtl.createSet($mod.TEnum.Red, $mod.TEnum.Blue)) $mod.e2 = $mod.e;',
  6317. 'for (var $l in $mod.s){',
  6318. ' $mod.e = +$l;',
  6319. ' $mod.e2 = $mod.e;',
  6320. '};',
  6321. 'for ($mod.er = 1; $mod.er <= 2; $mod.er++) ;',
  6322. '']));
  6323. end;
  6324. procedure TTestModule.TestNestBegin;
  6325. begin
  6326. StartProgram(false);
  6327. Add('begin');
  6328. Add(' begin');
  6329. Add(' begin');
  6330. Add(' end;');
  6331. Add(' begin');
  6332. Add(' if true then ;');
  6333. Add(' end;');
  6334. Add(' end;');
  6335. ConvertProgram;
  6336. CheckSource('TestNestBegin',
  6337. '',
  6338. 'if (true) ;');
  6339. end;
  6340. procedure TTestModule.TestUnitImplVars;
  6341. begin
  6342. StartUnit(false);
  6343. Add('interface');
  6344. Add('implementation');
  6345. Add('var');
  6346. Add(' V1:longint;');
  6347. Add(' V2:longint = 3;');
  6348. Add(' V3:string = ''abc'';');
  6349. ConvertUnit;
  6350. CheckSource('TestUnitImplVars',
  6351. LinesToStr([ // statements
  6352. 'var $impl = $mod.$impl;',
  6353. '']),
  6354. '', // this.$init
  6355. LinesToStr([ // implementation
  6356. '$impl.V1 = 0;',
  6357. '$impl.V2 = 3;',
  6358. '$impl.V3 = "abc";',
  6359. '']) );
  6360. end;
  6361. procedure TTestModule.TestUnitImplConsts;
  6362. begin
  6363. StartUnit(false);
  6364. Add('interface');
  6365. Add('implementation');
  6366. Add('const');
  6367. Add(' v1 = 3;');
  6368. Add(' v2:longint = 4;');
  6369. Add(' v3:string = ''abc'';');
  6370. ConvertUnit;
  6371. CheckSource('TestUnitImplConsts',
  6372. LinesToStr([ // statements
  6373. 'var $impl = $mod.$impl;',
  6374. '']),
  6375. '', // this.$init
  6376. LinesToStr([ // implementation
  6377. '$impl.v1 = 3;',
  6378. '$impl.v2 = 4;',
  6379. '$impl.v3 = "abc";',
  6380. '']) );
  6381. end;
  6382. procedure TTestModule.TestUnitImplRecord;
  6383. begin
  6384. StartUnit(false);
  6385. Add('interface');
  6386. Add('implementation');
  6387. Add('type');
  6388. Add(' TMyRecord = record');
  6389. Add(' i: longint;');
  6390. Add(' end;');
  6391. Add('var aRec: TMyRecord;');
  6392. Add('initialization');
  6393. Add(' arec.i:=3;');
  6394. ConvertUnit;
  6395. CheckSource('TestUnitImplRecord',
  6396. LinesToStr([ // statements
  6397. 'var $impl = $mod.$impl;',
  6398. '']),
  6399. // this.$init
  6400. '$impl.aRec.i = 3;',
  6401. LinesToStr([ // implementation
  6402. 'rtl.recNewT($impl, "TMyRecord", function () {',
  6403. ' this.i = 0;',
  6404. ' this.$eq = function (b) {',
  6405. ' return this.i === b.i;',
  6406. ' };',
  6407. ' this.$assign = function (s) {',
  6408. ' this.i = s.i;',
  6409. ' return this;',
  6410. ' };',
  6411. '});',
  6412. '$impl.aRec = $impl.TMyRecord.$new();',
  6413. '']) );
  6414. end;
  6415. procedure TTestModule.TestRenameJSNameConflict;
  6416. begin
  6417. StartProgram(false);
  6418. Add('var apply: longint;');
  6419. Add('var bind: longint;');
  6420. Add('var call: longint;');
  6421. Add('begin');
  6422. ConvertProgram;
  6423. CheckSource('TestRenameJSNameConflict',
  6424. LinesToStr([ // statements
  6425. 'this.Apply = 0;',
  6426. 'this.Bind = 0;',
  6427. 'this.Call = 0;'
  6428. ]),
  6429. LinesToStr([ // this.$main
  6430. ''
  6431. ]));
  6432. end;
  6433. procedure TTestModule.TestLocalConst;
  6434. begin
  6435. StartProgram(false);
  6436. Add('procedure DoIt;');
  6437. Add('const');
  6438. Add(' cA: longint = 1;');
  6439. Add(' cB = 2;');
  6440. Add(' procedure Sub;');
  6441. Add(' const');
  6442. Add(' csA = 3;');
  6443. Add(' cB: double = 4;');
  6444. Add(' begin');
  6445. Add(' cb:=cb+csa;');
  6446. Add(' ca:=ca+csa+5;');
  6447. Add(' end;');
  6448. Add('begin');
  6449. Add(' ca:=ca+cb+6;');
  6450. Add('end;');
  6451. Add('begin');
  6452. ConvertProgram;
  6453. CheckSource('TestLocalConst',
  6454. LinesToStr([
  6455. 'var cA = 1;',
  6456. 'var cB = 2;',
  6457. 'var csA = 3;',
  6458. 'var cB$1 = 4;',
  6459. 'this.DoIt = function () {',
  6460. ' function Sub() {',
  6461. ' cB$1 = cB$1 + 3;',
  6462. ' cA = cA + 3 + 5;',
  6463. ' };',
  6464. ' cA = cA + 2 + 6;',
  6465. '};'
  6466. ]),
  6467. LinesToStr([
  6468. ]));
  6469. end;
  6470. procedure TTestModule.TestVarExternal;
  6471. begin
  6472. StartProgram(false);
  6473. Add('var');
  6474. Add(' NaN: double; external name ''Global.NaN'';');
  6475. Add(' d: double;');
  6476. Add('begin');
  6477. Add(' d:=NaN;');
  6478. ConvertProgram;
  6479. CheckSource('TestVarExternal',
  6480. LinesToStr([
  6481. 'this.d = 0.0;'
  6482. ]),
  6483. LinesToStr([
  6484. '$mod.d = Global.NaN;'
  6485. ]));
  6486. end;
  6487. procedure TTestModule.TestVarExternalOtherUnit;
  6488. begin
  6489. AddModuleWithIntfImplSrc('unit2.pas',
  6490. LinesToStr([
  6491. 'var NaN: double; external name ''Global.NaN'';',
  6492. 'var iV: longint;'
  6493. ]),
  6494. '');
  6495. StartUnit(true);
  6496. Add('interface');
  6497. Add('uses unit2;');
  6498. Add('implementation');
  6499. Add('var');
  6500. Add(' d: double;');
  6501. Add(' i: longint; external name ''$i'';');
  6502. Add('begin');
  6503. Add(' d:=nan;');
  6504. Add(' d:=uNit2.nan;');
  6505. Add(' d:=test1.d;');
  6506. Add(' i:=iv;');
  6507. Add(' i:=uNit2.iv;');
  6508. Add(' i:=test1.i;');
  6509. ConvertUnit;
  6510. CheckSource('TestVarExternalOtherUnit',
  6511. LinesToStr([
  6512. 'var $impl = $mod.$impl;',
  6513. '']),
  6514. LinesToStr([ // this.$init
  6515. '$impl.d = Global.NaN;',
  6516. '$impl.d = Global.NaN;',
  6517. '$impl.d = $impl.d;',
  6518. '$i = pas.unit2.iV;',
  6519. '$i = pas.unit2.iV;',
  6520. '$i = $i;',
  6521. '']),
  6522. LinesToStr([ // implementation
  6523. '$impl.d = 0.0;',
  6524. '']) );
  6525. end;
  6526. procedure TTestModule.TestVarAbsoluteFail;
  6527. begin
  6528. StartProgram(false);
  6529. Add([
  6530. 'var',
  6531. ' a: longint;',
  6532. ' b: longword absolute a;',
  6533. 'begin']);
  6534. SetExpectedPasResolverError('Invalid variable modifier "absolute"',nInvalidVariableModifier);
  6535. ConvertProgram;
  6536. end;
  6537. procedure TTestModule.TestConstExternal;
  6538. begin
  6539. StartProgram(false);
  6540. Add([
  6541. 'const',
  6542. ' PI: double; external name ''Global.PI'';',
  6543. ' Tau = 2*pi;',
  6544. 'var d: double;',
  6545. 'begin',
  6546. ' d:=pi;',
  6547. ' d:=tau+pi;']);
  6548. ConvertProgram;
  6549. CheckSource('TestConstExternal',
  6550. LinesToStr([
  6551. 'this.Tau = 2*Global.PI;',
  6552. 'this.d = 0.0;'
  6553. ]),
  6554. LinesToStr([
  6555. '$mod.d = Global.PI;',
  6556. '$mod.d = $mod.Tau + Global.PI;'
  6557. ]));
  6558. end;
  6559. procedure TTestModule.TestDouble;
  6560. begin
  6561. StartProgram(false);
  6562. Add([
  6563. 'type',
  6564. ' TDateTime = double;',
  6565. 'const',
  6566. ' a = TDateTime(2.7);',
  6567. ' b = a + TDateTime(1.7);',
  6568. ' c = 0.9 + 0.1;',
  6569. ' f0_1 = 0.1;',
  6570. ' f0_3 = 0.3;',
  6571. ' fn0_1 = -0.1;',
  6572. ' fn0_3 = -0.3;',
  6573. ' fn0_003 = -0.003;',
  6574. ' fn0_123456789 = -0.123456789;',
  6575. ' fn300_0 = -300.0;',
  6576. ' fn123456_0 = -123456.0;',
  6577. ' fn1234567_8 = -1234567.8;',
  6578. ' fn12345678_9 = -12345678.9;',
  6579. ' f1_0En12 = 1E-12;',
  6580. ' fn1_0En12 = -1E-12;',
  6581. ' maxdouble = 1.7e+308;',
  6582. ' mindouble = -1.7e+308;',
  6583. ' MinSafeIntDouble = -$1fffffffffffff;',
  6584. ' MinSafeIntDouble2 = -$20000000000000-1;',
  6585. ' MaxSafeIntDouble = $1fffffffffffff;',
  6586. ' DZeroResolution = 1E-12;',
  6587. ' Minus1 = -1E-12;',
  6588. ' EPS = 1E-9;',
  6589. ' DELTA = 0.001;',
  6590. ' Big = 129.789E+100;',
  6591. ' Test0_15 = 0.15;',
  6592. ' Test999 = 2.9999999999999;',
  6593. ' Test111999 = 211199999999999000.0;',
  6594. ' TestMinus111999 = -211199999999999000.0;',
  6595. 'var',
  6596. ' d: double = b;',
  6597. 'begin',
  6598. ' d:=1.0;',
  6599. ' d:=1.0/3.0;',
  6600. ' d:=1/3;',
  6601. ' d:=5.0E-324;',
  6602. ' d:=1.7E308;',
  6603. ' d:=001.00E00;',
  6604. ' d:=002.00E001;',
  6605. ' d:=003.000E000;',
  6606. ' d:=-004.00E-00;',
  6607. ' d:=-005.00E-001;',
  6608. ' d:=10**3;',
  6609. ' d:=10 mod 3;',
  6610. ' d:=10 div 3;',
  6611. ' d:=c;',
  6612. ' d:=f0_1;',
  6613. ' d:=f0_3;',
  6614. ' d:=fn0_1;',
  6615. ' d:=fn0_3;',
  6616. ' d:=fn0_003;',
  6617. ' d:=fn0_123456789;',
  6618. ' d:=fn300_0;',
  6619. ' d:=fn123456_0;',
  6620. ' d:=fn1234567_8;',
  6621. ' d:=fn12345678_9;',
  6622. ' d:=f1_0En12;',
  6623. ' d:=fn1_0En12;',
  6624. ' d:=maxdouble;',
  6625. ' d:=mindouble;',
  6626. ' d:=MinSafeIntDouble;',
  6627. ' d:=double(MinSafeIntDouble);',
  6628. ' d:=MinSafeIntDouble2;',
  6629. ' d:=double(MinSafeIntDouble2);',
  6630. ' d:=MaxSafeIntDouble;',
  6631. ' d:=default(double);',
  6632. '']);
  6633. ConvertProgram;
  6634. CheckSource('TestDouble',
  6635. LinesToStr([
  6636. 'this.a = 2.7;',
  6637. 'this.b = 2.7 + 1.7;',
  6638. 'this.c = 0.9 + 0.1;',
  6639. 'this.f0_1 = 0.1;',
  6640. 'this.f0_3 = 0.3;',
  6641. 'this.fn0_1 = -0.1;',
  6642. 'this.fn0_3 = -0.3;',
  6643. 'this.fn0_003 = -0.003;',
  6644. 'this.fn0_123456789 = -0.123456789;',
  6645. 'this.fn300_0 = -300.0;',
  6646. 'this.fn123456_0 = -123456.0;',
  6647. 'this.fn1234567_8 = -1234567.8;',
  6648. 'this.fn12345678_9 = -12345678.9;',
  6649. 'this.f1_0En12 = 1E-12;',
  6650. 'this.fn1_0En12 = -1E-12;',
  6651. 'this.maxdouble = 1.7e+308;',
  6652. 'this.mindouble = -1.7e+308;',
  6653. 'this.MinSafeIntDouble = -0x1fffffffffffff;',
  6654. 'this.MinSafeIntDouble2 = -0x20000000000000 - 1;',
  6655. 'this.MaxSafeIntDouble = 0x1fffffffffffff;',
  6656. 'this.DZeroResolution = 1E-12;',
  6657. 'this.Minus1 = -1E-12;',
  6658. 'this.EPS = 1E-9;',
  6659. 'this.DELTA = 0.001;',
  6660. 'this.Big = 129.789E+100;',
  6661. 'this.Test0_15 = 0.15;',
  6662. 'this.Test999 = 2.9999999999999;',
  6663. 'this.Test111999 = 211199999999999000.0;',
  6664. 'this.TestMinus111999 = -211199999999999000.0;',
  6665. 'this.d = 4.4;'
  6666. ]),
  6667. LinesToStr([
  6668. '$mod.d = 1.0;',
  6669. '$mod.d = 1.0 / 3.0;',
  6670. '$mod.d = 1 / 3;',
  6671. '$mod.d = 5.0E-324;',
  6672. '$mod.d = 1.7E308;',
  6673. '$mod.d = 1.00E0;',
  6674. '$mod.d = 2.00E1;',
  6675. '$mod.d = 3.000E0;',
  6676. '$mod.d = -4.00E-0;',
  6677. '$mod.d = -5.00E-1;',
  6678. '$mod.d = Math.pow(10, 3);',
  6679. '$mod.d = 10 % 3;',
  6680. '$mod.d = rtl.trunc(10 / 3);',
  6681. '$mod.d = 1;',
  6682. '$mod.d = 0.1;',
  6683. '$mod.d = 0.3;',
  6684. '$mod.d = -0.1;',
  6685. '$mod.d = -0.3;',
  6686. '$mod.d = -0.003;',
  6687. '$mod.d = -0.123456789;',
  6688. '$mod.d = -300;',
  6689. '$mod.d = -123456;',
  6690. '$mod.d = -1234567.8;',
  6691. '$mod.d = -1.23456789E7;',
  6692. '$mod.d = 1E-12;',
  6693. '$mod.d = -1E-12;',
  6694. '$mod.d = 1.7E308;',
  6695. '$mod.d = -1.7E308;',
  6696. '$mod.d = -9007199254740991;',
  6697. '$mod.d = -9007199254740991;',
  6698. '$mod.d = -9.007199254740992E15;',
  6699. '$mod.d = -9.007199254740992E15;',
  6700. '$mod.d = 9007199254740991;',
  6701. '$mod.d = 0.0;',
  6702. '']));
  6703. end;
  6704. procedure TTestModule.TestInteger;
  6705. begin
  6706. StartProgram(false);
  6707. Add([
  6708. 'const',
  6709. ' MinInt = low(NativeInt);',
  6710. ' MaxInt = high(NativeInt);',
  6711. 'type',
  6712. ' {#TMyInt}TMyInt = MinInt..MaxInt;',
  6713. 'const',
  6714. ' a = low(TMyInt)+High(TMyInt);',
  6715. 'var',
  6716. ' i: TMyInt;',
  6717. 'begin',
  6718. ' i:=-MinInt;',
  6719. ' i:=default(TMyInt);',
  6720. ' i:=low(i)+high(i);',
  6721. '']);
  6722. ConvertProgram;
  6723. CheckSource('TestIntegerRange',
  6724. LinesToStr([
  6725. 'this.MinInt = -9007199254740991;',
  6726. 'this.MaxInt = 9007199254740991;',
  6727. 'this.a = -9007199254740991 + 9007199254740991;',
  6728. 'this.i = 0;',
  6729. '']),
  6730. LinesToStr([
  6731. '$mod.i = - -9007199254740991;',
  6732. '$mod.i = -9007199254740991;',
  6733. '$mod.i = -9007199254740991 + 9007199254740991;',
  6734. '']));
  6735. end;
  6736. procedure TTestModule.TestIntegerRange;
  6737. begin
  6738. StartProgram(false);
  6739. Add([
  6740. 'const',
  6741. ' MinInt = -1;',
  6742. ' MaxInt = +1;',
  6743. 'type',
  6744. ' {#TMyInt}TMyInt = MinInt..MaxInt;',
  6745. ' TInt2 = 1..3;',
  6746. 'const',
  6747. ' a = low(TMyInt)+High(TMyInt);',
  6748. ' b = low(TInt2)+High(TInt2);',
  6749. ' s1 = [1];',
  6750. ' s2 = [1,2];',
  6751. ' s3 = [1..3];',
  6752. ' s4 = [low(shortint)..high(shortint)];',
  6753. ' s5 = [succ(low(shortint))..pred(high(shortint))];',
  6754. ' s6 = 1 in s2;',
  6755. 'var',
  6756. ' i: TMyInt;',
  6757. ' i2: TInt2;',
  6758. 'begin',
  6759. ' i:=i2;',
  6760. ' i:=default(TMyInt);',
  6761. ' if i=i2 then ;']);
  6762. ConvertProgram;
  6763. CheckSource('TestIntegerRange',
  6764. LinesToStr([
  6765. 'this.MinInt = -1;',
  6766. 'this.MaxInt = +1;',
  6767. 'this.a = -1 + 1;',
  6768. 'this.b = 1 + 3;',
  6769. 'this.s1 = rtl.createSet(1);',
  6770. 'this.s2 = rtl.createSet(1, 2);',
  6771. 'this.s3 = rtl.createSet(null, 1, 3);',
  6772. 'this.s4 = rtl.createSet(null, -128, 127);',
  6773. 'this.s5 = rtl.createSet(null, -128 + 1, 127 - 1);',
  6774. 'this.s6 = 1 in this.s2;',
  6775. 'this.i = 0;',
  6776. 'this.i2 = 0;',
  6777. '']),
  6778. LinesToStr([
  6779. '$mod.i = $mod.i2;',
  6780. '$mod.i = -1;',
  6781. 'if ($mod.i === $mod.i2) ;',
  6782. '']));
  6783. end;
  6784. procedure TTestModule.TestIntegerTypecasts;
  6785. begin
  6786. StartProgram(false);
  6787. Add([
  6788. 'var',
  6789. ' i: nativeint;',
  6790. ' b: byte;',
  6791. ' sh: shortint;',
  6792. ' w: word;',
  6793. ' sm: smallint;',
  6794. ' lw: longword;',
  6795. ' li: longint;',
  6796. 'begin',
  6797. ' b:=byte(i);',
  6798. ' sh:=shortint(i);',
  6799. ' w:=word(i);',
  6800. ' sm:=smallint(i);',
  6801. ' lw:=longword(i);',
  6802. ' li:=longint(i);',
  6803. '']);
  6804. ConvertProgram;
  6805. CheckSource('TestIntegerTypecasts',
  6806. LinesToStr([
  6807. 'this.i = 0;',
  6808. 'this.b = 0;',
  6809. 'this.sh = 0;',
  6810. 'this.w = 0;',
  6811. 'this.sm = 0;',
  6812. 'this.lw = 0;',
  6813. 'this.li = 0;',
  6814. '']),
  6815. LinesToStr([
  6816. '$mod.b = $mod.i & 255;',
  6817. '$mod.sh = (($mod.i & 255) << 24) >> 24;',
  6818. '$mod.w = $mod.i & 65535;',
  6819. '$mod.sm = (($mod.i & 65535) << 16) >> 16;',
  6820. '$mod.lw = $mod.i >>> 0;',
  6821. '$mod.li = $mod.i & 0xFFFFFFFF;',
  6822. '']));
  6823. end;
  6824. procedure TTestModule.TestInteger_BitwiseShrNativeInt;
  6825. begin
  6826. StartProgram(false);
  6827. Add([
  6828. 'var',
  6829. ' i,j: nativeint;',
  6830. 'begin',
  6831. ' i:=i shr 0;',
  6832. ' i:=i shr 1;',
  6833. ' i:=i shr 3;',
  6834. ' i:=i shr 54;',
  6835. ' i:=j shr i;',
  6836. '']);
  6837. ConvertProgram;
  6838. CheckResolverUnexpectedHints;
  6839. CheckSource('TestInteger_BitwiseShrNativeInt',
  6840. LinesToStr([
  6841. 'this.i = 0;',
  6842. 'this.j = 0;',
  6843. '']),
  6844. LinesToStr([
  6845. '$mod.i = $mod.i;',
  6846. '$mod.i = Math.floor($mod.i / 2);',
  6847. '$mod.i = Math.floor($mod.i / 8);',
  6848. '$mod.i = 0;',
  6849. '$mod.i = rtl.shr($mod.j, $mod.i);',
  6850. '']));
  6851. end;
  6852. procedure TTestModule.TestInteger_BitwiseShlNativeInt;
  6853. begin
  6854. StartProgram(false);
  6855. Add([
  6856. 'var',
  6857. ' i: nativeint;',
  6858. 'begin',
  6859. ' i:=i shl 0;',
  6860. ' i:=i shl 54;',
  6861. ' i:=123456789012 shl 1;',
  6862. ' i:=i shl 1;',
  6863. '']);
  6864. ConvertProgram;
  6865. CheckResolverUnexpectedHints;
  6866. CheckSource('TestInteger_BitwiseShrNativeInt',
  6867. LinesToStr([
  6868. 'this.i = 0;',
  6869. '']),
  6870. LinesToStr([
  6871. '$mod.i = $mod.i;',
  6872. '$mod.i = 0;',
  6873. '$mod.i = 246913578024;',
  6874. '$mod.i = rtl.shl($mod.i, 1);',
  6875. '']));
  6876. end;
  6877. procedure TTestModule.TestInteger_SystemFunc;
  6878. begin
  6879. StartProgram(true);
  6880. Add([
  6881. 'var',
  6882. ' i: byte;',
  6883. ' s: string;',
  6884. 'begin',
  6885. ' system.inc(i);',
  6886. ' system.str(i,s);',
  6887. ' s:=system.str(i);',
  6888. ' i:=system.low(i);',
  6889. ' i:=system.high(i);',
  6890. ' i:=system.pred(i);',
  6891. ' i:=system.succ(i);',
  6892. '']);
  6893. ConvertProgram;
  6894. CheckResolverUnexpectedHints;
  6895. CheckSource('TestInteger_SystemFunc',
  6896. LinesToStr([
  6897. 'this.i = 0;',
  6898. 'this.s = "";',
  6899. '']),
  6900. LinesToStr([
  6901. '$mod.i += 1;',
  6902. '$mod.s = "" + $mod.i;',
  6903. '$mod.s = "" + $mod.i;',
  6904. '$mod.i = 0;',
  6905. '$mod.i = 255;',
  6906. '$mod.i = $mod.i - 1;',
  6907. '$mod.i = $mod.i + 1;',
  6908. '']));
  6909. end;
  6910. procedure TTestModule.TestCurrency;
  6911. begin
  6912. StartProgram(false);
  6913. Add([
  6914. 'type',
  6915. ' TCoin = currency;',
  6916. 'const',
  6917. ' a = TCoin(2.7);',
  6918. ' b = a + TCoin(1.7);',
  6919. ' MinSafeIntCurrency: TCoin = -92233720368.5477;',
  6920. ' MaxSafeIntCurrency: TCoin = 92233720368.5477;',
  6921. 'var',
  6922. ' c: TCoin = b;',
  6923. ' i: nativeint;',
  6924. ' d: double;',
  6925. ' j: jsvalue;',
  6926. 'function DoIt(c: currency): currency; begin end;',
  6927. 'function GetIt(d: double): double; begin end;',
  6928. 'procedure Write(v: jsvalue); begin end;',
  6929. 'begin',
  6930. ' c:=1.0;',
  6931. ' c:=0.1;',
  6932. ' c:=1.0/3.0;',
  6933. ' c:=1/3;',
  6934. ' c:=a;',
  6935. ' d:=c;',
  6936. ' c:=d;',
  6937. ' c:=currency(c);',
  6938. ' c:=currency(d);',
  6939. ' d:=double(c);',
  6940. ' c:=i;',
  6941. ' c:=currency(i);',
  6942. //' i:=c;', not allowed
  6943. ' i:=nativeint(c);',
  6944. ' c:=c+a;',
  6945. ' c:=-c-a;',
  6946. ' c:=d+c;',
  6947. ' c:=c+d;',
  6948. ' c:=d-c;',
  6949. ' c:=c-d;',
  6950. ' c:=c*a;',
  6951. ' c:=a*c;',
  6952. ' c:=d*c;',
  6953. ' c:=c*d;',
  6954. ' c:=c/a;',
  6955. ' c:=a/c;',
  6956. ' c:=d/c;',
  6957. ' c:=c/d;',
  6958. ' c:=c**a;',
  6959. ' c:=a**c;',
  6960. ' c:=d**c;',
  6961. ' c:=c**d;',
  6962. ' if c=c then ;',
  6963. ' if c=a then ;',
  6964. ' if a=c then ;',
  6965. ' if d=c then ;',
  6966. ' if c=d then ;',
  6967. ' c:=DoIt(c);',
  6968. ' c:=DoIt(i);',
  6969. ' c:=DoIt(d);',
  6970. ' c:=GetIt(c);',
  6971. ' j:=c;',
  6972. ' Write(c);',
  6973. ' c:=default(currency);',
  6974. ' j:=str(c);',
  6975. ' j:=str(c:0:3);',
  6976. '']);
  6977. ConvertProgram;
  6978. CheckSource('TestCurrency',
  6979. LinesToStr([
  6980. 'this.a = 27000;',
  6981. 'this.b = this.a + 17000;',
  6982. 'this.MinSafeIntCurrency = -92233720368.5477;',
  6983. 'this.MaxSafeIntCurrency = 92233720368.5477;',
  6984. 'this.c = this.b;',
  6985. 'this.i = 0;',
  6986. 'this.d = 0.0;',
  6987. 'this.j = undefined;',
  6988. 'this.DoIt = function (c) {',
  6989. ' var Result = 0;',
  6990. ' return Result;',
  6991. '};',
  6992. 'this.GetIt = function (d) {',
  6993. ' var Result = 0.0;',
  6994. ' return Result;',
  6995. '};',
  6996. 'this.Write = function (v) {',
  6997. '};',
  6998. '']),
  6999. LinesToStr([
  7000. '$mod.c = 10000;',
  7001. '$mod.c = 1000;',
  7002. '$mod.c = rtl.trunc((1.0 / 3.0) * 10000);',
  7003. '$mod.c = rtl.trunc((1 / 3) * 10000);',
  7004. '$mod.c = $mod.a;',
  7005. '$mod.d = $mod.c / 10000;',
  7006. '$mod.c = rtl.trunc($mod.d * 10000);',
  7007. '$mod.c = $mod.c;',
  7008. '$mod.c = $mod.d * 10000;',
  7009. '$mod.d = $mod.c / 10000;',
  7010. '$mod.c = $mod.i * 10000;',
  7011. '$mod.c = $mod.i * 10000;',
  7012. '$mod.i = rtl.trunc($mod.c / 10000);',
  7013. '$mod.c = $mod.c + $mod.a;',
  7014. '$mod.c = -$mod.c - $mod.a;',
  7015. '$mod.c = ($mod.d * 10000) + $mod.c;',
  7016. '$mod.c = $mod.c + ($mod.d * 10000);',
  7017. '$mod.c = ($mod.d * 10000) - $mod.c;',
  7018. '$mod.c = $mod.c - ($mod.d * 10000);',
  7019. '$mod.c = ($mod.c * $mod.a) / 10000;',
  7020. '$mod.c = ($mod.a * $mod.c) / 10000;',
  7021. '$mod.c = $mod.d * $mod.c;',
  7022. '$mod.c = $mod.c * $mod.d;',
  7023. '$mod.c = rtl.trunc(($mod.c / $mod.a) * 10000);',
  7024. '$mod.c = rtl.trunc(($mod.a / $mod.c) * 10000);',
  7025. '$mod.c = rtl.trunc($mod.d / $mod.c);',
  7026. '$mod.c = rtl.trunc($mod.c / $mod.d);',
  7027. '$mod.c = rtl.trunc(Math.pow($mod.c / 10000, $mod.a / 10000) * 10000);',
  7028. '$mod.c = rtl.trunc(Math.pow($mod.a / 10000, $mod.c / 10000) * 10000);',
  7029. '$mod.c = rtl.trunc(Math.pow($mod.d, $mod.c / 10000) * 10000);',
  7030. '$mod.c = rtl.trunc(Math.pow($mod.c / 10000, $mod.d) * 10000);',
  7031. 'if ($mod.c === $mod.c) ;',
  7032. 'if ($mod.c === $mod.a) ;',
  7033. 'if ($mod.a === $mod.c) ;',
  7034. 'if (($mod.d * 10000) === $mod.c) ;',
  7035. 'if ($mod.c === ($mod.d * 10000)) ;',
  7036. '$mod.c = $mod.DoIt($mod.c);',
  7037. '$mod.c = $mod.DoIt($mod.i * 10000);',
  7038. '$mod.c = $mod.DoIt($mod.d * 10000);',
  7039. '$mod.c = rtl.trunc($mod.GetIt($mod.c / 10000) * 10000);',
  7040. '$mod.j = $mod.c / 10000;',
  7041. '$mod.Write($mod.c / 10000);',
  7042. '$mod.c = 0;',
  7043. '$mod.j = rtl.floatToStr($mod.c / 10000);',
  7044. '$mod.j = rtl.floatToStr($mod.c / 10000, 0, 3);',
  7045. '']));
  7046. end;
  7047. procedure TTestModule.TestForBoolDo;
  7048. begin
  7049. StartProgram(false);
  7050. Add([
  7051. 'var b: boolean;',
  7052. 'begin',
  7053. ' for b:=false to true do ;',
  7054. ' for b:=b downto false do ;',
  7055. ' for b in boolean do ;',
  7056. '']);
  7057. ConvertProgram;
  7058. CheckSource('TestForBoolDo',
  7059. LinesToStr([ // statements
  7060. 'this.b = false;']),
  7061. LinesToStr([ // this.$main
  7062. 'for (var $l = 0; $l <= 1; $l++) $mod.b = $l !== 0;',
  7063. 'for (var $l1 = +$mod.b; $l1 >= 0; $l1--) $mod.b = $l1 !== 0;',
  7064. 'for (var $l2 = 0; $l2 <= 1; $l2++) $mod.b = $l2 !== 0;',
  7065. '']));
  7066. end;
  7067. procedure TTestModule.TestForIntDo;
  7068. begin
  7069. StartProgram(false);
  7070. Add([
  7071. 'var i: longint;',
  7072. 'begin',
  7073. ' for i:=3 to 5 do ;',
  7074. ' for i:=i downto 2 do ;',
  7075. ' for i in byte do ;',
  7076. '']);
  7077. ConvertProgram;
  7078. CheckSource('TestForIntDo',
  7079. LinesToStr([ // statements
  7080. 'this.i = 0;']),
  7081. LinesToStr([ // this.$main
  7082. 'for ($mod.i = 3; $mod.i <= 5; $mod.i++) ;',
  7083. 'for (var $l = $mod.i; $l >= 2; $l--) $mod.i = $l;',
  7084. 'for (var $l1 = 0; $l1 <= 255; $l1++) $mod.i = $l1;',
  7085. '']));
  7086. end;
  7087. procedure TTestModule.TestForIntInDo;
  7088. begin
  7089. StartProgram(false);
  7090. Add([
  7091. 'type',
  7092. ' TSetOfInt = set of byte;',
  7093. ' TIntRg = 3..7;',
  7094. ' TSetOfIntRg = set of TIntRg;',
  7095. 'var',
  7096. ' i,i2: longint;',
  7097. ' a1: array of byte;',
  7098. ' a2: array[1..3] of byte;',
  7099. ' soi: TSetOfInt;',
  7100. ' soir: TSetOfIntRg;',
  7101. ' ir: TIntRg;',
  7102. 'begin',
  7103. ' for i in byte do ;',
  7104. ' for i in a1 do ;',
  7105. ' for i in a2 do ;',
  7106. ' for i in [11..13] do ;',
  7107. ' for i in TSetOfInt do ;',
  7108. ' for i in TIntRg do ;',
  7109. ' for i in soi do i2:=i;',
  7110. ' for i in TSetOfIntRg do ;',
  7111. ' for i in soir do ;',
  7112. ' for ir in TIntRg do ;',
  7113. ' for ir in TSetOfIntRg do ;',
  7114. ' for ir in soir do ;',
  7115. '']);
  7116. ConvertProgram;
  7117. CheckSource('TestForIntInDo',
  7118. LinesToStr([ // statements
  7119. 'this.i = 0;',
  7120. 'this.i2 = 0;',
  7121. 'this.a1 = [];',
  7122. 'this.a2 = rtl.arraySetLength(null, 0, 3);',
  7123. 'this.soi = {};',
  7124. 'this.soir = {};',
  7125. 'this.ir = 0;',
  7126. '']),
  7127. LinesToStr([ // this.$main
  7128. 'for (var $l = 0; $l <= 255; $l++) $mod.i = $l;',
  7129. 'for (var $in = $mod.a1, $l1 = 0, $end = rtl.length($in) - 1; $l1 <= $end; $l1++) $mod.i = $in[$l1];',
  7130. 'for (var $in1 = $mod.a2, $l2 = 0, $end1 = rtl.length($in1) - 1; $l2 <= $end1; $l2++) $mod.i = $in1[$l2];',
  7131. 'for (var $l3 = 11; $l3 <= 13; $l3++) $mod.i = $l3;',
  7132. 'for (var $l4 = 0; $l4 <= 255; $l4++) $mod.i = $l4;',
  7133. 'for (var $l5 = 3; $l5 <= 7; $l5++) $mod.i = $l5;',
  7134. 'for (var $l6 in $mod.soi) {',
  7135. ' $mod.i = +$l6;',
  7136. ' $mod.i2 = $mod.i;',
  7137. '};',
  7138. 'for (var $l7 = 3; $l7 <= 7; $l7++) $mod.i = $l7;',
  7139. 'for (var $l8 in $mod.soir) $mod.i = +$l8;',
  7140. 'for (var $l9 = 3; $l9 <= 7; $l9++) $mod.ir = $l9;',
  7141. 'for (var $l10 = 3; $l10 <= 7; $l10++) $mod.ir = $l10;',
  7142. 'for (var $l11 in $mod.soir) $mod.ir = +$l11;',
  7143. '']));
  7144. end;
  7145. procedure TTestModule.TestCharConst;
  7146. begin
  7147. StartProgram(false);
  7148. Add([
  7149. 'const',
  7150. ' a = #$00F3;',
  7151. ' c: char = ''1'';',
  7152. ' wc: widechar = ''ä'';',
  7153. 'begin',
  7154. ' c:=#0;',
  7155. ' c:=#1;',
  7156. ' c:=#9;',
  7157. ' c:=#10;',
  7158. ' c:=#13;',
  7159. ' c:=#31;',
  7160. ' c:=#32;',
  7161. ' c:=#$A;',
  7162. ' c:=#$0A;',
  7163. ' c:=#$b;',
  7164. ' c:=#$0b;',
  7165. ' c:=^A;',
  7166. ' c:=''"'';',
  7167. ' c:=default(char);',
  7168. ' c:=#$00E4;', // ä
  7169. ' c:=''ä'';',
  7170. ' c:=#$E4;', // ä
  7171. ' c:=#$D800;', // invalid UTF-16
  7172. ' c:=#$DFFF;', // invalid UTF-16
  7173. ' c:=#$FFFF;', // last UCS-2
  7174. ' c:=high(c);', // last UCS-2
  7175. '']);
  7176. ConvertProgram;
  7177. CheckSource('TestCharConst',
  7178. LinesToStr([
  7179. 'this.a="ó";',
  7180. 'this.c="1";',
  7181. 'this.wc="ä";'
  7182. ]),
  7183. LinesToStr([
  7184. '$mod.c="\x00";',
  7185. '$mod.c="\x01";',
  7186. '$mod.c="\t";',
  7187. '$mod.c="\n";',
  7188. '$mod.c="\r";',
  7189. '$mod.c="\x1F";',
  7190. '$mod.c=" ";',
  7191. '$mod.c="\n";',
  7192. '$mod.c="\n";',
  7193. '$mod.c="\x0B";',
  7194. '$mod.c="\x0B";',
  7195. '$mod.c="\x01";',
  7196. '$mod.c=''"'';',
  7197. '$mod.c="\x00";',
  7198. '$mod.c = "ä";',
  7199. '$mod.c = "ä";',
  7200. '$mod.c = "ä";',
  7201. '$mod.c="\uD800";',
  7202. '$mod.c="\uDFFF";',
  7203. '$mod.c="\uFFFF";',
  7204. '$mod.c="\uFFFF";',
  7205. '']));
  7206. end;
  7207. procedure TTestModule.TestChar_Compare;
  7208. begin
  7209. StartProgram(false);
  7210. Add('var');
  7211. Add(' c: char;');
  7212. Add(' b: boolean;');
  7213. Add('begin');
  7214. Add(' b:=c=''1'';');
  7215. Add(' b:=''2''=c;');
  7216. Add(' b:=''3''=''4'';');
  7217. Add(' b:=c<>''5'';');
  7218. Add(' b:=''6''<>c;');
  7219. Add(' b:=c>''7'';');
  7220. Add(' b:=''8''>c;');
  7221. Add(' b:=c>=''9'';');
  7222. Add(' b:=''A''>=c;');
  7223. Add(' b:=c<''B'';');
  7224. Add(' b:=''C''<c;');
  7225. Add(' b:=c<=''D'';');
  7226. Add(' b:=''E''<=c;');
  7227. ConvertProgram;
  7228. CheckSource('TestChar_Compare',
  7229. LinesToStr([
  7230. 'this.c="";',
  7231. 'this.b = false;'
  7232. ]),
  7233. LinesToStr([
  7234. '$mod.b = $mod.c === "1";',
  7235. '$mod.b = "2" === $mod.c;',
  7236. '$mod.b = "3" === "4";',
  7237. '$mod.b = $mod.c !== "5";',
  7238. '$mod.b = "6" !== $mod.c;',
  7239. '$mod.b = $mod.c > "7";',
  7240. '$mod.b = "8" > $mod.c;',
  7241. '$mod.b = $mod.c >= "9";',
  7242. '$mod.b = "A" >= $mod.c;',
  7243. '$mod.b = $mod.c < "B";',
  7244. '$mod.b = "C" < $mod.c;',
  7245. '$mod.b = $mod.c <= "D";',
  7246. '$mod.b = "E" <= $mod.c;',
  7247. '']));
  7248. end;
  7249. procedure TTestModule.TestChar_BuiltInProcs;
  7250. begin
  7251. StartProgram(false);
  7252. Add([
  7253. 'var',
  7254. ' c: char;',
  7255. ' i: longint;',
  7256. ' s: string;',
  7257. 'begin',
  7258. ' i:=ord(c);',
  7259. ' i:=ord(s[i]);',
  7260. ' c:=chr(i);',
  7261. ' c:=pred(c);',
  7262. ' c:=succ(c);',
  7263. ' c:=low(c);',
  7264. ' c:=high(c);',
  7265. ' i:=byte(c);',
  7266. ' i:=word(c);',
  7267. ' i:=longint(c);',
  7268. '']);
  7269. ConvertProgram;
  7270. CheckSource('TestChar_BuiltInProcs',
  7271. LinesToStr([
  7272. 'this.c = "";',
  7273. 'this.i = 0;',
  7274. 'this.s = "";'
  7275. ]),
  7276. LinesToStr([
  7277. '$mod.i = $mod.c.charCodeAt();',
  7278. '$mod.i = $mod.s.charCodeAt($mod.i-1);',
  7279. '$mod.c = String.fromCharCode($mod.i);',
  7280. '$mod.c = String.fromCharCode($mod.c.charCodeAt() - 1);',
  7281. '$mod.c = String.fromCharCode($mod.c.charCodeAt() + 1);',
  7282. '$mod.c = "\x00";',
  7283. '$mod.c = "\uFFFF";',
  7284. '$mod.i = $mod.c.charCodeAt() & 255;',
  7285. '$mod.i = $mod.c.charCodeAt();',
  7286. '$mod.i = $mod.c.charCodeAt() & 0xFFFFFFFF;',
  7287. '']));
  7288. end;
  7289. procedure TTestModule.TestStringConst;
  7290. begin
  7291. StartProgram(false);
  7292. Add([
  7293. '{$H+}',
  7294. 'const',
  7295. ' a = #$00F3#$017C;', // first <256, then >=256
  7296. ' b = string(''a'');',
  7297. ' c = string(''ä'');',
  7298. ' d = UnicodeString(''b'');',
  7299. ' e = UnicodeString(''ö'');',
  7300. 'var',
  7301. ' s: string = ''abc'';',
  7302. 'begin',
  7303. ' s:='''';',
  7304. ' s:=#13#10;',
  7305. ' s:=#9''foo'';',
  7306. ' s:=#$A9;',
  7307. ' s:=''foo''#13''bar'';',
  7308. ' s:=''"'';',
  7309. ' s:=''"''''"'';',
  7310. ' s:=#$20AC;', // euro
  7311. ' s:=#$10437;', // outside BMP
  7312. ' s:=default(string);',
  7313. ' s:=concat(s);',
  7314. ' s:=concat(s,''a'',s)',
  7315. '']);
  7316. ConvertProgram;
  7317. CheckSource('TestStringConst',
  7318. LinesToStr([
  7319. 'this.a = "óż";',
  7320. 'this.b = "a";',
  7321. 'this.c = "ä";',
  7322. 'this.d = "b";',
  7323. 'this.e = "ö";',
  7324. 'this.s="abc";',
  7325. '']),
  7326. LinesToStr([
  7327. '$mod.s="";',
  7328. '$mod.s="\r\n";',
  7329. '$mod.s="\tfoo";',
  7330. '$mod.s="©";',
  7331. '$mod.s="foo\rbar";',
  7332. '$mod.s=''"'';',
  7333. '$mod.s=''"\''"'';',
  7334. '$mod.s="€";',
  7335. '$mod.s="'#$F0#$90#$90#$B7'";',
  7336. '$mod.s="";',
  7337. '$mod.s = $mod.s;',
  7338. '$mod.s = $mod.s.concat("a", $mod.s);',
  7339. '']));
  7340. end;
  7341. procedure TTestModule.TestStringConstSurrogate;
  7342. begin
  7343. StartProgram(false);
  7344. Add([
  7345. 'var',
  7346. ' s: string;',
  7347. 'begin',
  7348. ' s:=''😊'';', // 1F60A
  7349. '']);
  7350. ConvertProgram;
  7351. CheckSource('TestStringConstSurrogate',
  7352. LinesToStr([
  7353. 'this.s="";'
  7354. ]),
  7355. LinesToStr([
  7356. '$mod.s="😊";'
  7357. ]));
  7358. end;
  7359. procedure TTestModule.TestString_Length;
  7360. begin
  7361. StartProgram(false);
  7362. Add('const c = ''foo'';');
  7363. Add('var');
  7364. Add(' s: string;');
  7365. Add(' i: longint;');
  7366. Add('begin');
  7367. Add(' i:=length(s);');
  7368. Add(' i:=length(s+s);');
  7369. Add(' i:=length(''abc'');');
  7370. Add(' i:=length(c);');
  7371. ConvertProgram;
  7372. CheckSource('TestString_Length',
  7373. LinesToStr([
  7374. 'this.c = "foo";',
  7375. 'this.s = "";',
  7376. 'this.i = 0;',
  7377. '']),
  7378. LinesToStr([
  7379. '$mod.i = $mod.s.length;',
  7380. '$mod.i = ($mod.s+$mod.s).length;',
  7381. '$mod.i = "abc".length;',
  7382. '$mod.i = $mod.c.length;',
  7383. '']));
  7384. end;
  7385. procedure TTestModule.TestString_Compare;
  7386. begin
  7387. StartProgram(false);
  7388. Add('var');
  7389. Add(' s, t: string;');
  7390. Add(' b: boolean;');
  7391. Add('begin');
  7392. Add(' b:=s=t;');
  7393. Add(' b:=s<>t;');
  7394. Add(' b:=s>t;');
  7395. Add(' b:=s>=t;');
  7396. Add(' b:=s<t;');
  7397. Add(' b:=s<=t;');
  7398. ConvertProgram;
  7399. CheckSource('TestString_Compare',
  7400. LinesToStr([ // statements
  7401. 'this.s = "";',
  7402. 'this.t = "";',
  7403. 'this.b =false;'
  7404. ]),
  7405. LinesToStr([ // this.$main
  7406. '$mod.b = $mod.s === $mod.t;',
  7407. '$mod.b = $mod.s !== $mod.t;',
  7408. '$mod.b = $mod.s > $mod.t;',
  7409. '$mod.b = $mod.s >= $mod.t;',
  7410. '$mod.b = $mod.s < $mod.t;',
  7411. '$mod.b = $mod.s <= $mod.t;',
  7412. '']));
  7413. end;
  7414. procedure TTestModule.TestString_SetLength;
  7415. begin
  7416. StartProgram(false);
  7417. Add([
  7418. 'procedure DoIt(var s: string);',
  7419. 'begin',
  7420. ' SetLength(s,2);',
  7421. 'end;',
  7422. 'var s: string;',
  7423. 'begin',
  7424. ' SetLength(s,3);',
  7425. '']);
  7426. ConvertProgram;
  7427. CheckSource('TestString_SetLength',
  7428. LinesToStr([ // statements
  7429. 'this.DoIt = function (s) {',
  7430. ' s.set(rtl.strSetLength(s.get(), 2));',
  7431. '};',
  7432. 'this.s = "";',
  7433. '']),
  7434. LinesToStr([ // this.$main
  7435. '$mod.s = rtl.strSetLength($mod.s, 3);'
  7436. ]));
  7437. end;
  7438. procedure TTestModule.TestString_CharAt;
  7439. begin
  7440. StartProgram(false);
  7441. Add([
  7442. 'var',
  7443. ' s: string;',
  7444. ' c: char;',
  7445. ' b: boolean;',
  7446. 'begin',
  7447. ' b:= s[1] = c;',
  7448. ' b:= c = s[1];',
  7449. ' b:= c <> s[1];',
  7450. ' b:= c > s[1];',
  7451. ' b:= c >= s[1];',
  7452. ' b:= c < s[2];',
  7453. ' b:= c <= s[1];',
  7454. ' s[1] := c;',
  7455. ' s[2+3] := c;']);
  7456. ConvertProgram;
  7457. CheckSource('TestString_CharAt',
  7458. LinesToStr([ // statements
  7459. 'this.s = "";',
  7460. 'this.c = "";',
  7461. 'this.b = false;'
  7462. ]),
  7463. LinesToStr([ // this.$main
  7464. '$mod.b = $mod.s.charAt(0) === $mod.c;',
  7465. '$mod.b = $mod.c === $mod.s.charAt(0);',
  7466. '$mod.b = $mod.c !== $mod.s.charAt(0);',
  7467. '$mod.b = $mod.c > $mod.s.charAt(0);',
  7468. '$mod.b = $mod.c >= $mod.s.charAt(0);',
  7469. '$mod.b = $mod.c < $mod.s.charAt(1);',
  7470. '$mod.b = $mod.c <= $mod.s.charAt(0);',
  7471. '$mod.s = rtl.setCharAt($mod.s, 0, $mod.c);',
  7472. '$mod.s = rtl.setCharAt($mod.s, (2 + 3) - 1, $mod.c);',
  7473. '']));
  7474. end;
  7475. procedure TTestModule.TestStringHMinusFail;
  7476. begin
  7477. StartProgram(false);
  7478. Add([
  7479. '{$H-}',
  7480. 'var s: string;',
  7481. 'begin']);
  7482. ConvertProgram;
  7483. CheckHint(mtWarning,nWarnIllegalCompilerDirectiveX,'Warning: test1.pp(3,6) : Illegal compiler directive "H-"');
  7484. end;
  7485. procedure TTestModule.TestStr;
  7486. begin
  7487. StartProgram(false);
  7488. Add('var');
  7489. Add(' b: boolean;');
  7490. Add(' i: longint;');
  7491. Add(' d: double;');
  7492. Add(' s: string;');
  7493. Add('begin');
  7494. Add(' str(b,s);');
  7495. Add(' str(i,s);');
  7496. Add(' str(d,s);');
  7497. Add(' str(i:3,s);');
  7498. Add(' str(d:3:2,s);');
  7499. Add(' Str(12.456:12:1,s);');
  7500. Add(' Str(12.456:12,s);');
  7501. Add(' s:=str(b);');
  7502. Add(' s:=str(i);');
  7503. Add(' s:=str(d);');
  7504. Add(' s:=str(i,i);');
  7505. Add(' s:=str(i:3);');
  7506. Add(' s:=str(d:3:2);');
  7507. Add(' s:=str(i:4,i);');
  7508. Add(' s:=str(i,i:5);');
  7509. Add(' s:=str(i:4,i:5);');
  7510. Add(' s:=str(s,s);');
  7511. Add(' s:=str(s,''foo'');');
  7512. ConvertProgram;
  7513. CheckSource('TestStr',
  7514. LinesToStr([ // statements
  7515. 'this.b = false;',
  7516. 'this.i = 0;',
  7517. 'this.d = 0.0;',
  7518. 'this.s = "";',
  7519. '']),
  7520. LinesToStr([ // this.$main
  7521. '$mod.s = ""+$mod.b;',
  7522. '$mod.s = ""+$mod.i;',
  7523. '$mod.s = rtl.floatToStr($mod.d);',
  7524. '$mod.s = rtl.spaceLeft(""+$mod.i,3);',
  7525. '$mod.s = rtl.floatToStr($mod.d,3,2);',
  7526. '$mod.s = rtl.floatToStr(12.456,12,1);',
  7527. '$mod.s = rtl.floatToStr(12.456,12);',
  7528. '$mod.s = ""+$mod.b;',
  7529. '$mod.s = ""+$mod.i;',
  7530. '$mod.s = rtl.floatToStr($mod.d);',
  7531. '$mod.s = ""+$mod.i+$mod.i;',
  7532. '$mod.s = rtl.spaceLeft(""+$mod.i,3);',
  7533. '$mod.s = rtl.floatToStr($mod.d,3,2);',
  7534. '$mod.s = rtl.spaceLeft("" + $mod.i, 4) + $mod.i;',
  7535. '$mod.s = "" + $mod.i + rtl.spaceLeft("" + $mod.i, 5);',
  7536. '$mod.s = rtl.spaceLeft("" + $mod.i, 4) + rtl.spaceLeft("" + $mod.i, 5);',
  7537. '$mod.s = $mod.s + $mod.s;',
  7538. '$mod.s = $mod.s + "foo";',
  7539. '']));
  7540. end;
  7541. procedure TTestModule.TestBaseType_AnsiStringFail;
  7542. begin
  7543. StartProgram(false);
  7544. Add('var s: AnsiString');
  7545. SetExpectedPasResolverError('identifier not found "AnsiString"',PasResolveEval.nIdentifierNotFound);
  7546. ConvertProgram;
  7547. end;
  7548. procedure TTestModule.TestBaseType_WideStringFail;
  7549. begin
  7550. StartProgram(false);
  7551. Add('var s: WideString');
  7552. SetExpectedPasResolverError('identifier not found "WideString"',PasResolveEval.nIdentifierNotFound);
  7553. ConvertProgram;
  7554. end;
  7555. procedure TTestModule.TestBaseType_ShortStringFail;
  7556. begin
  7557. StartProgram(false);
  7558. Add('var s: ShortString');
  7559. SetExpectedPasResolverError('identifier not found "ShortString"',PasResolveEval.nIdentifierNotFound);
  7560. ConvertProgram;
  7561. end;
  7562. procedure TTestModule.TestBaseType_RawByteStringFail;
  7563. begin
  7564. StartProgram(false);
  7565. Add('var s: RawByteString');
  7566. SetExpectedPasResolverError('identifier not found "RawByteString"',PasResolveEval.nIdentifierNotFound);
  7567. ConvertProgram;
  7568. end;
  7569. procedure TTestModule.TestTypeShortstring_Fail;
  7570. begin
  7571. StartProgram(false);
  7572. Add('type t = string[12];');
  7573. Add('var s: t;');
  7574. Add('begin');
  7575. SetExpectedPasResolverError('illegal qualifier "["',nIllegalQualifier);
  7576. ConvertProgram;
  7577. end;
  7578. procedure TTestModule.TestCharSet_Custom;
  7579. begin
  7580. StartProgram(false);
  7581. Add([
  7582. 'type',
  7583. ' TCharRg = ''a''..''z'';',
  7584. ' TSetOfCharRg = set of TCharRg;',
  7585. ' TCharRg2 = ''m''..''p'';',
  7586. 'const',
  7587. ' crg: TCharRg = ''b'';',
  7588. 'var',
  7589. ' c: char;',
  7590. ' crg2: TCharRg2;',
  7591. ' s: TSetOfCharRg;',
  7592. 'begin',
  7593. ' c:=crg;',
  7594. ' crg:=c;',
  7595. ' crg2:=crg;',
  7596. ' if c=crg then ;',
  7597. ' if crg=c then ;',
  7598. ' if crg=crg2 then ;',
  7599. ' if c in s then ;',
  7600. ' if crg2 in s then ;',
  7601. ' c:=default(TCharRg);',
  7602. '']);
  7603. ConvertProgram;
  7604. CheckSource('TestCharSet_Custom',
  7605. LinesToStr([ // statements
  7606. 'this.crg = "b";',
  7607. 'this.c = "";',
  7608. 'this.crg2 = "m";',
  7609. 'this.s = {};',
  7610. '']),
  7611. LinesToStr([ // this.$main
  7612. '$mod.c = $mod.crg;',
  7613. '$mod.crg = $mod.c;',
  7614. '$mod.crg2 = $mod.crg;',
  7615. 'if ($mod.c === $mod.crg) ;',
  7616. 'if ($mod.crg === $mod.c) ;',
  7617. 'if ($mod.crg === $mod.crg2) ;',
  7618. 'if ($mod.c.charCodeAt() in $mod.s) ;',
  7619. 'if ($mod.crg2.charCodeAt() in $mod.s) ;',
  7620. '$mod.c = "a";',
  7621. '']));
  7622. end;
  7623. procedure TTestModule.TestWideChar_VarArg;
  7624. begin
  7625. StartProgram(false);
  7626. Add([
  7627. 'procedure Fly(var c: char);',
  7628. 'begin',
  7629. 'end;',
  7630. 'procedure Run(var c: widechar);',
  7631. 'begin',
  7632. 'end;',
  7633. 'var',
  7634. ' c: char;',
  7635. ' wc: widechar;',
  7636. 'begin',
  7637. ' Fly(wc);',
  7638. ' Run(c);',
  7639. '']);
  7640. ConvertProgram;
  7641. CheckSource('TestWideChar_VarArg',
  7642. LinesToStr([ // statements
  7643. 'this.Fly = function (c) {',
  7644. '};',
  7645. 'this.Run = function (c) {',
  7646. '};',
  7647. 'this.c = "";',
  7648. 'this.wc = "";',
  7649. '']),
  7650. LinesToStr([ // this.$main
  7651. '$mod.Fly({',
  7652. ' p: $mod,',
  7653. ' get: function () {',
  7654. ' return this.p.wc;',
  7655. ' },',
  7656. ' set: function (v) {',
  7657. ' this.p.wc = v;',
  7658. ' }',
  7659. '});',
  7660. '$mod.Run({',
  7661. ' p: $mod,',
  7662. ' get: function () {',
  7663. ' return this.p.c;',
  7664. ' },',
  7665. ' set: function (v) {',
  7666. ' this.p.c = v;',
  7667. ' }',
  7668. '});',
  7669. '',
  7670. '']));
  7671. end;
  7672. procedure TTestModule.TestForCharDo;
  7673. begin
  7674. StartProgram(false);
  7675. Add([
  7676. 'var c: char;',
  7677. 'begin',
  7678. ' for c:=''a'' to ''c'' do ;',
  7679. ' for c:=c downto ''a'' do ;',
  7680. ' for c:=''Б'' to ''Я'' do ;',
  7681. '']);
  7682. ConvertProgram;
  7683. CheckSource('TestForCharDo',
  7684. LinesToStr([ // statements
  7685. 'this.c = "";']),
  7686. LinesToStr([ // this.$main
  7687. 'for (var $l = 97; $l <= 99; $l++) $mod.c = String.fromCharCode($l);',
  7688. 'for (var $l1 = $mod.c.charCodeAt(); $l1 >= 97; $l1--) $mod.c = String.fromCharCode($l1);',
  7689. 'for (var $l2 = 1041; $l2 <= 1071; $l2++) $mod.c = String.fromCharCode($l2);',
  7690. '']));
  7691. end;
  7692. procedure TTestModule.TestForCharInDo;
  7693. begin
  7694. StartProgram(false);
  7695. Add([
  7696. 'type',
  7697. ' TSetOfChar = set of char;',
  7698. ' TCharRg = ''a''..''z'';',
  7699. ' TSetOfCharRg = set of TCharRg;',
  7700. 'const Foo = ''foo'';',
  7701. 'var',
  7702. ' c,c2: char;',
  7703. ' s: string;',
  7704. ' a1: array of char;',
  7705. ' a2: array[1..3] of char;',
  7706. ' soc: TSetOfChar;',
  7707. ' socr: TSetOfCharRg;',
  7708. ' cr: TCharRg;',
  7709. 'begin',
  7710. ' for c in foo do ;',
  7711. ' for c in s do ;',
  7712. ' for c in char do ;',
  7713. ' for c in a1 do ;',
  7714. ' for c in a2 do ;',
  7715. ' for c in [''1''..''3''] do ;',
  7716. ' for c in TSetOfChar do ;',
  7717. ' for c in TCharRg do ;',
  7718. ' for c in soc do c2:=c;',
  7719. ' for c in TSetOfCharRg do ;',
  7720. ' for c in socr do ;',
  7721. ' for cr in TCharRg do ;',
  7722. ' for cr in TSetOfCharRg do ;',
  7723. ' for cr in socr do ;',
  7724. '']);
  7725. ConvertProgram;
  7726. CheckSource('TestForCharInDo',
  7727. LinesToStr([ // statements
  7728. 'this.Foo = "foo";',
  7729. 'this.c = "";',
  7730. 'this.c2 = "";',
  7731. 'this.s = "";',
  7732. 'this.a1 = [];',
  7733. 'this.a2 = rtl.arraySetLength(null, "", 3);',
  7734. 'this.soc = {};',
  7735. 'this.socr = {};',
  7736. 'this.cr = "a";',
  7737. '']),
  7738. LinesToStr([ // this.$main
  7739. 'for (var $in = $mod.Foo, $l = 0, $end = $in.length - 1; $l <= $end; $l++) $mod.c = $in.charAt($l);',
  7740. 'for (var $in1 = $mod.s, $l1 = 0, $end1 = $in1.length - 1; $l1 <= $end1; $l1++) $mod.c = $in1.charAt($l1);',
  7741. 'for (var $l2 = 0; $l2 <= 65535; $l2++) $mod.c = String.fromCharCode($l2);',
  7742. 'for (var $in2 = $mod.a1, $l3 = 0, $end2 = rtl.length($in2) - 1; $l3 <= $end2; $l3++) $mod.c = $in2[$l3];',
  7743. 'for (var $in3 = $mod.a2, $l4 = 0, $end3 = rtl.length($in3) - 1; $l4 <= $end3; $l4++) $mod.c = $in3[$l4];',
  7744. 'for (var $l5 = 49; $l5 <= 51; $l5++) $mod.c = String.fromCharCode($l5);',
  7745. 'for (var $l6 = 0; $l6 <= 65535; $l6++) $mod.c = String.fromCharCode($l6);',
  7746. 'for (var $l7 = 97; $l7 <= 122; $l7++) $mod.c = String.fromCharCode($l7);',
  7747. 'for (var $l8 in $mod.soc) {',
  7748. ' $mod.c = String.fromCharCode($l8);',
  7749. ' $mod.c2 = $mod.c;',
  7750. '};',
  7751. 'for (var $l9 = 97; $l9 <= 122; $l9++) $mod.c = String.fromCharCode($l9);',
  7752. 'for (var $l10 in $mod.socr) $mod.c = String.fromCharCode($l10);',
  7753. 'for (var $l11 = 97; $l11 <= 122; $l11++) $mod.cr = String.fromCharCode($l11);',
  7754. 'for (var $l12 = 97; $l12 <= 122; $l12++) $mod.cr = String.fromCharCode($l12);',
  7755. 'for (var $l13 in $mod.socr) $mod.cr = String.fromCharCode($l13);',
  7756. '']));
  7757. end;
  7758. procedure TTestModule.TestProcTwoArgs;
  7759. begin
  7760. StartProgram(false);
  7761. Add('procedure Test(a,b: longint);');
  7762. Add('begin');
  7763. Add('end;');
  7764. Add('begin');
  7765. ConvertProgram;
  7766. CheckSource('TestProcTwoArgs',
  7767. LinesToStr([ // statements
  7768. 'this.Test = function (a,b) {',
  7769. '};'
  7770. ]),
  7771. LinesToStr([ // this.$main
  7772. ''
  7773. ]));
  7774. end;
  7775. procedure TTestModule.TestProc_DefaultValue;
  7776. begin
  7777. StartProgram(false);
  7778. Add('procedure p1(i: longint = 1);');
  7779. Add('begin');
  7780. Add('end;');
  7781. Add('procedure p2(i: longint = 1; c: char = ''a'');');
  7782. Add('begin');
  7783. Add('end;');
  7784. Add('procedure p3(d: double = 1.0; b: boolean = false; s: string = ''abc'');');
  7785. Add('begin');
  7786. Add('end;');
  7787. Add('begin');
  7788. Add(' p1;');
  7789. Add(' p1();');
  7790. Add(' p1(11);');
  7791. Add(' p2;');
  7792. Add(' p2();');
  7793. Add(' p2(12);');
  7794. Add(' p2(13,''b'');');
  7795. Add(' p3();');
  7796. ConvertProgram;
  7797. CheckSource('TestProc_DefaultValue',
  7798. LinesToStr([ // statements
  7799. 'this.p1 = function (i) {',
  7800. '};',
  7801. 'this.p2 = function (i,c) {',
  7802. '};',
  7803. 'this.p3 = function (d,b,s) {',
  7804. '};'
  7805. ]),
  7806. LinesToStr([ // this.$main
  7807. ' $mod.p1(1);',
  7808. ' $mod.p1(1);',
  7809. ' $mod.p1(11);',
  7810. ' $mod.p2(1,"a");',
  7811. ' $mod.p2(1,"a");',
  7812. ' $mod.p2(12,"a");',
  7813. ' $mod.p2(13,"b");',
  7814. ' $mod.p3(1.0,false,"abc");'
  7815. ]));
  7816. end;
  7817. procedure TTestModule.TestFunctionInt;
  7818. begin
  7819. StartProgram(false);
  7820. Add('function MyTest(Bar: longint): longint;');
  7821. Add('begin');
  7822. Add(' Result:=2*bar');
  7823. Add('end;');
  7824. Add('begin');
  7825. ConvertProgram;
  7826. CheckSource('TestFunctionInt',
  7827. LinesToStr([ // statements
  7828. 'this.MyTest = function (Bar) {',
  7829. ' var Result = 0;',
  7830. ' Result = 2*Bar;',
  7831. ' return Result;',
  7832. '};'
  7833. ]),
  7834. LinesToStr([ // this.$main
  7835. ''
  7836. ]));
  7837. end;
  7838. procedure TTestModule.TestFunctionString;
  7839. begin
  7840. StartProgram(false);
  7841. Add('function Test(Bar: string): string;');
  7842. Add('begin');
  7843. Add(' Result:=bar+BAR');
  7844. Add('end;');
  7845. Add('begin');
  7846. ConvertProgram;
  7847. CheckSource('TestFunctionString',
  7848. LinesToStr([ // statements
  7849. 'this.Test = function (Bar) {',
  7850. ' var Result = "";',
  7851. ' Result = Bar+Bar;',
  7852. ' return Result;',
  7853. '};'
  7854. ]),
  7855. LinesToStr([ // this.$main
  7856. ''
  7857. ]));
  7858. end;
  7859. procedure TTestModule.TestIfThen;
  7860. begin
  7861. StartProgram(false);
  7862. Add([
  7863. 'var b: boolean;',
  7864. 'begin',
  7865. ' if b then ;',
  7866. ' if b then else ;']);
  7867. ConvertProgram;
  7868. CheckSource('TestIfThen',
  7869. LinesToStr([ // statements
  7870. 'this.b = false;',
  7871. '']),
  7872. LinesToStr([ // this.$main
  7873. 'if ($mod.b) ;',
  7874. 'if ($mod.b) ;',
  7875. '']));
  7876. end;
  7877. procedure TTestModule.TestForLoop;
  7878. begin
  7879. StartProgram(false);
  7880. Add('var');
  7881. Add(' vI, vJ, vN: longint;');
  7882. Add('begin');
  7883. Add(' VJ:=0;');
  7884. Add(' VN:=3;');
  7885. Add(' for VI:=1 to VN do');
  7886. Add(' begin');
  7887. Add(' VJ:=VJ+VI;');
  7888. Add(' end;');
  7889. ConvertProgram;
  7890. CheckSource('TestForLoop',
  7891. LinesToStr([ // statements
  7892. 'this.vI = 0;',
  7893. 'this.vJ = 0;',
  7894. 'this.vN = 0;'
  7895. ]),
  7896. LinesToStr([ // this.$main
  7897. ' $mod.vJ = 0;',
  7898. ' $mod.vN = 3;',
  7899. ' for (var $l = 1, $end = $mod.vN; $l <= $end; $l++) {',
  7900. ' $mod.vI = $l;',
  7901. ' $mod.vJ = $mod.vJ + $mod.vI;',
  7902. ' };',
  7903. '']));
  7904. end;
  7905. procedure TTestModule.TestForLoopInsideFunction;
  7906. begin
  7907. StartProgram(false);
  7908. Add('function SumNumbers(Count: longint): longint;');
  7909. Add('var');
  7910. Add(' vI, vJ: longint;');
  7911. Add('begin');
  7912. Add(' vj:=0;');
  7913. Add(' for vi:=1 to count do');
  7914. Add(' begin');
  7915. Add(' vj:=vj+vi;');
  7916. Add(' end;');
  7917. Add('end;');
  7918. Add('begin');
  7919. Add(' sumnumbers(3);');
  7920. ConvertProgram;
  7921. CheckSource('TestForLoopInsideFunction',
  7922. LinesToStr([ // statements
  7923. 'this.SumNumbers = function (Count) {',
  7924. ' var Result = 0;',
  7925. ' var vI = 0;',
  7926. ' var vJ = 0;',
  7927. ' vJ = 0;',
  7928. ' for (var $l = 1, $end = Count; $l <= $end; $l++) {',
  7929. ' vI = $l;',
  7930. ' vJ = vJ + vI;',
  7931. ' };',
  7932. ' return Result;',
  7933. '};'
  7934. ]),
  7935. LinesToStr([ // $mod.$main
  7936. ' $mod.SumNumbers(3);'
  7937. ]));
  7938. end;
  7939. procedure TTestModule.TestForLoop_ReadVarAfter;
  7940. begin
  7941. StartProgram(false);
  7942. Add('var');
  7943. Add(' vI: longint;');
  7944. Add('begin');
  7945. Add(' for vi:=1 to 2 do ;');
  7946. Add(' if vi=3 then ;');
  7947. ConvertProgram;
  7948. CheckSource('TestForLoop',
  7949. LinesToStr([ // statements
  7950. 'this.vI = 0;'
  7951. ]),
  7952. LinesToStr([ // this.$main
  7953. ' for ($mod.vI = 1; $mod.vI <= 2; $mod.vI++) ;',
  7954. ' if ($mod.vI===3) ;'
  7955. ]));
  7956. end;
  7957. procedure TTestModule.TestForLoop_Nested;
  7958. begin
  7959. StartProgram(false);
  7960. Add('function SumNumbers(Count: longint): longint;');
  7961. Add('var');
  7962. Add(' vI, vJ, vK: longint;');
  7963. Add('begin');
  7964. Add(' VK:=0;');
  7965. Add(' for VI:=1 to count do');
  7966. Add(' begin');
  7967. Add(' for vj:=1 to vi do');
  7968. Add(' begin');
  7969. Add(' vk:=VK+VI;');
  7970. Add(' end;');
  7971. Add(' end;');
  7972. Add('end;');
  7973. Add('begin');
  7974. Add(' sumnumbers(3);');
  7975. ConvertProgram;
  7976. CheckSource('TestForLoopInFunction',
  7977. LinesToStr([ // statements
  7978. 'this.SumNumbers = function (Count) {',
  7979. ' var Result = 0;',
  7980. ' var vI = 0;',
  7981. ' var vJ = 0;',
  7982. ' var vK = 0;',
  7983. ' vK = 0;',
  7984. ' for (var $l = 1, $end = Count; $l <= $end; $l++) {',
  7985. ' vI = $l;',
  7986. ' for (var $l1 = 1, $end1 = vI; $l1 <= $end1; $l1++) {',
  7987. ' vJ = $l1;',
  7988. ' vK = vK + vI;',
  7989. ' };',
  7990. ' };',
  7991. ' return Result;',
  7992. '};'
  7993. ]),
  7994. LinesToStr([ // $mod.$main
  7995. ' $mod.SumNumbers(3);'
  7996. ]));
  7997. end;
  7998. procedure TTestModule.TestRepeatUntil;
  7999. begin
  8000. StartProgram(false);
  8001. Add('var');
  8002. Add(' vI, vJ, vN: longint;');
  8003. Add('begin');
  8004. Add(' vn:=3;');
  8005. Add(' vj:=0;');
  8006. Add(' VI:=0;');
  8007. Add(' repeat');
  8008. Add(' VI:=vi+1;');
  8009. Add(' vj:=VJ+vI;');
  8010. Add(' until vi>=vn');
  8011. ConvertProgram;
  8012. CheckSource('TestRepeatUntil',
  8013. LinesToStr([ // statements
  8014. 'this.vI = 0;',
  8015. 'this.vJ = 0;',
  8016. 'this.vN = 0;'
  8017. ]),
  8018. LinesToStr([ // $mod.$main
  8019. ' $mod.vN = 3;',
  8020. ' $mod.vJ = 0;',
  8021. ' $mod.vI = 0;',
  8022. ' do{',
  8023. ' $mod.vI = $mod.vI + 1;',
  8024. ' $mod.vJ = $mod.vJ + $mod.vI;',
  8025. ' }while(!($mod.vI>=$mod.vN));'
  8026. ]));
  8027. end;
  8028. procedure TTestModule.TestAsmBlock;
  8029. begin
  8030. StartProgram(false);
  8031. Add([
  8032. 'var',
  8033. ' vI: longint;',
  8034. 'begin',
  8035. ' vi:=1;',
  8036. ' asm',
  8037. ' if (vI===1) {',
  8038. ' vI=2;',
  8039. //' console.log(''end;'');', ToDo
  8040. ' }',
  8041. ' if (vI===2){ vI=3; }',
  8042. ' end;',
  8043. ' VI:=4;']);
  8044. ConvertProgram;
  8045. CheckSource('TestAsmBlock',
  8046. LinesToStr([ // statements
  8047. 'this.vI = 0;'
  8048. ]),
  8049. LinesToStr([ // $mod.$main
  8050. '$mod.vI = 1;',
  8051. 'if (vI===1) {',
  8052. ' vI=2;',
  8053. '}',
  8054. 'if (vI===2){ vI=3; }',
  8055. ';',
  8056. '$mod.vI = 4;'
  8057. ]));
  8058. end;
  8059. procedure TTestModule.TestAsmPas_Impl;
  8060. begin
  8061. StartUnit(false);
  8062. Add('interface');
  8063. Add('const cIntf: longint = 1;');
  8064. Add('var vIntf: longint;');
  8065. Add('implementation');
  8066. Add('const cImpl: longint = 2;');
  8067. Add('var vImpl: longint;');
  8068. Add('procedure DoIt;');
  8069. Add('const cLoc: longint = 3;');
  8070. Add('var vLoc: longint;');
  8071. Add('begin;');
  8072. Add(' asm');
  8073. //Add(' pas(vIntf)=pas(cIntf);');
  8074. //Add(' pas(vImpl)=pas(cImpl);');
  8075. //Add(' pas(vLoc)=pas(cLoc);');
  8076. Add(' end;');
  8077. Add('end;');
  8078. ConvertUnit;
  8079. CheckSource('TestAsmPas_Impl',
  8080. LinesToStr([
  8081. 'var $impl = $mod.$impl;',
  8082. 'this.cIntf = 1;',
  8083. 'this.vIntf = 0;',
  8084. '']),
  8085. '', // this.$init
  8086. LinesToStr([ // implementation
  8087. '$impl.cImpl = 2;',
  8088. '$impl.vImpl = 0;',
  8089. 'var cLoc = 3;',
  8090. '$impl.DoIt = function () {',
  8091. ' var vLoc = 0;',
  8092. '};',
  8093. '']) );
  8094. end;
  8095. procedure TTestModule.TestTryFinally;
  8096. begin
  8097. StartProgram(false);
  8098. Add('var i: longint;');
  8099. Add('begin');
  8100. Add(' try');
  8101. Add(' i:=0; i:=2 div i;');
  8102. Add(' finally');
  8103. Add(' i:=3');
  8104. Add(' end;');
  8105. ConvertProgram;
  8106. CheckSource('TestTryFinally',
  8107. LinesToStr([ // statements
  8108. 'this.i = 0;'
  8109. ]),
  8110. LinesToStr([ // $mod.$main
  8111. 'try {',
  8112. ' $mod.i = 0;',
  8113. ' $mod.i = rtl.trunc(2 / $mod.i);',
  8114. '} finally {',
  8115. ' $mod.i = 3;',
  8116. '};'
  8117. ]));
  8118. end;
  8119. procedure TTestModule.TestTryExcept;
  8120. begin
  8121. StartProgram(false);
  8122. Add([
  8123. 'type',
  8124. ' TObject = class end;',
  8125. ' Exception = class Msg: string; end;',
  8126. ' EInvalidCast = class(Exception) end;',
  8127. 'var vI: longint;',
  8128. 'begin',
  8129. ' try',
  8130. ' vi:=1;',
  8131. ' except',
  8132. ' vi:=2',
  8133. ' end;',
  8134. ' try',
  8135. ' vi:=3;',
  8136. ' except',
  8137. ' raise;',
  8138. ' end;',
  8139. ' try',
  8140. ' VI:=4;',
  8141. ' except',
  8142. ' on einvalidcast do',
  8143. ' raise;',
  8144. ' on E: exception do',
  8145. ' if e.msg='''' then',
  8146. ' raise e;',
  8147. ' else',
  8148. ' vi:=5',
  8149. ' end;',
  8150. ' try',
  8151. ' VI:=6;',
  8152. ' except',
  8153. ' on einvalidcast do ;',
  8154. ' end;',
  8155. '']);
  8156. ConvertProgram;
  8157. CheckSource('TestTryExcept',
  8158. LinesToStr([ // statements
  8159. 'rtl.createClass(this, "TObject", null, function () {',
  8160. ' this.$init = function () {',
  8161. ' };',
  8162. ' this.$final = function () {',
  8163. ' };',
  8164. '});',
  8165. 'rtl.createClass(this, "Exception", this.TObject, function () {',
  8166. ' this.$init = function () {',
  8167. ' $mod.TObject.$init.call(this);',
  8168. ' this.Msg = "";',
  8169. ' };',
  8170. '});',
  8171. 'rtl.createClass(this, "EInvalidCast", this.Exception, function () {',
  8172. '});',
  8173. 'this.vI = 0;'
  8174. ]),
  8175. LinesToStr([ // $mod.$main
  8176. 'try {',
  8177. ' $mod.vI = 1;',
  8178. '} catch ($e) {',
  8179. ' $mod.vI = 2;',
  8180. '};',
  8181. 'try {',
  8182. ' $mod.vI = 3;',
  8183. '} catch ($e) {',
  8184. ' throw $e;',
  8185. '};',
  8186. 'try {',
  8187. ' $mod.vI = 4;',
  8188. '} catch ($e) {',
  8189. ' if ($mod.EInvalidCast.isPrototypeOf($e)){',
  8190. ' throw $e',
  8191. ' } else if ($mod.Exception.isPrototypeOf($e)) {',
  8192. ' var E = $e;',
  8193. ' if (E.Msg === "") throw E;',
  8194. ' } else {',
  8195. ' $mod.vI = 5;',
  8196. ' }',
  8197. '};',
  8198. 'try {',
  8199. ' $mod.vI = 6;',
  8200. '} catch ($e) {',
  8201. ' if ($mod.EInvalidCast.isPrototypeOf($e)){' ,
  8202. ' } else throw $e',
  8203. '};',
  8204. '']));
  8205. end;
  8206. procedure TTestModule.TestTryExcept_ReservedWords;
  8207. begin
  8208. StartProgram(false);
  8209. Add([
  8210. 'type',
  8211. ' TObject = class end;',
  8212. ' Exception = class',
  8213. ' Symbol: string;',
  8214. ' end;',
  8215. 'var &try: longint;',
  8216. 'begin',
  8217. ' try',
  8218. ' &try:=4;',
  8219. ' except',
  8220. ' on Error: exception do',
  8221. ' if errOR.symBol='''' then',
  8222. ' raise ERRor;',
  8223. ' end;',
  8224. '']);
  8225. ConvertProgram;
  8226. CheckSource('TestTryExcept_ReservedWords',
  8227. LinesToStr([ // statements
  8228. 'rtl.createClass(this, "TObject", null, function () {',
  8229. ' this.$init = function () {',
  8230. ' };',
  8231. ' this.$final = function () {',
  8232. ' };',
  8233. '});',
  8234. 'rtl.createClass(this, "Exception", this.TObject, function () {',
  8235. ' this.$init = function () {',
  8236. ' $mod.TObject.$init.call(this);',
  8237. ' this.Symbol = "";',
  8238. ' };',
  8239. '});',
  8240. 'this.Try = 0;',
  8241. '']),
  8242. LinesToStr([ // $mod.$main
  8243. 'try {',
  8244. ' $mod.Try = 4;',
  8245. '} catch ($e) {',
  8246. ' if ($mod.Exception.isPrototypeOf($e)) {',
  8247. ' var error = $e;',
  8248. ' if (error.Symbol === "") throw error;',
  8249. ' } else throw $e',
  8250. '};',
  8251. '']));
  8252. end;
  8253. procedure TTestModule.TestIfThenRaiseElse;
  8254. begin
  8255. StartProgram(false);
  8256. Add([
  8257. 'type',
  8258. ' TObject = class',
  8259. ' constructor Create;',
  8260. ' end;',
  8261. 'constructor TObject.Create;',
  8262. 'begin',
  8263. 'end;',
  8264. 'var b: boolean;',
  8265. 'begin',
  8266. ' if b then',
  8267. ' raise TObject.Create',
  8268. ' else',
  8269. ' b:=false;',
  8270. '']);
  8271. ConvertProgram;
  8272. CheckSource('TestIfThenRaiseElse',
  8273. LinesToStr([ // statements
  8274. 'rtl.createClass(this, "TObject", null, function () {',
  8275. ' this.$init = function () {',
  8276. ' };',
  8277. ' this.$final = function () {',
  8278. ' };',
  8279. ' this.Create = function () {',
  8280. ' return this;',
  8281. ' };',
  8282. '});',
  8283. 'this.b = false;',
  8284. '']),
  8285. LinesToStr([ // $mod.$main
  8286. 'if ($mod.b) {',
  8287. ' throw $mod.TObject.$create("Create")}',
  8288. ' else $mod.b = false;',
  8289. '']));
  8290. end;
  8291. procedure TTestModule.TestCaseOf;
  8292. begin
  8293. StartProgram(false);
  8294. Add([
  8295. 'const e: longint; external name ''$e'';',
  8296. 'var vI: longint;',
  8297. 'begin',
  8298. ' case vi of',
  8299. ' 1: ;',
  8300. ' 2: vi:=3;',
  8301. ' e: ;',
  8302. ' else',
  8303. ' VI:=4',
  8304. ' end;']);
  8305. ConvertProgram;
  8306. CheckSource('TestCaseOf',
  8307. LinesToStr([ // statements
  8308. 'this.vI = 0;'
  8309. ]),
  8310. LinesToStr([ // $mod.$main
  8311. 'var $tmp = $mod.vI;',
  8312. 'if ($tmp === 1) {}',
  8313. 'else if ($tmp === 2) {',
  8314. ' $mod.vI = 3}',
  8315. ' else if ($tmp === $e) {}',
  8316. 'else {',
  8317. ' $mod.vI = 4;',
  8318. '};'
  8319. ]));
  8320. end;
  8321. procedure TTestModule.TestCaseOf_UseSwitch;
  8322. begin
  8323. StartProgram(false);
  8324. Converter.UseSwitchStatement:=true;
  8325. Add('var Vi: longint;');
  8326. Add('begin');
  8327. Add(' case vi of');
  8328. Add(' 1: ;');
  8329. Add(' 2: VI:=3;');
  8330. Add(' else');
  8331. Add(' vi:=4');
  8332. Add(' end;');
  8333. ConvertProgram;
  8334. CheckSource('TestCaseOf_UseSwitch',
  8335. LinesToStr([ // statements
  8336. 'this.Vi = 0;'
  8337. ]),
  8338. LinesToStr([ // $mod.$main
  8339. 'switch ($mod.Vi) {',
  8340. 'case 1:',
  8341. ' break;',
  8342. 'case 2:',
  8343. ' $mod.Vi = 3;',
  8344. ' break;',
  8345. 'default:',
  8346. ' $mod.Vi = 4;',
  8347. '};'
  8348. ]));
  8349. end;
  8350. procedure TTestModule.TestCaseOfNoElse;
  8351. begin
  8352. StartProgram(false);
  8353. Add('var Vi: longint;');
  8354. Add('begin');
  8355. Add(' case vi of');
  8356. Add(' 1: begin vi:=2; VI:=3; end;');
  8357. Add(' end;');
  8358. ConvertProgram;
  8359. CheckSource('TestCaseOfNoElse',
  8360. LinesToStr([ // statements
  8361. 'this.Vi = 0;'
  8362. ]),
  8363. LinesToStr([ // $mod.$main
  8364. 'var $tmp = $mod.Vi;',
  8365. 'if ($tmp === 1) {',
  8366. ' $mod.Vi = 2;',
  8367. ' $mod.Vi = 3;',
  8368. '};'
  8369. ]));
  8370. end;
  8371. procedure TTestModule.TestCaseOfNoElse_UseSwitch;
  8372. begin
  8373. StartProgram(false);
  8374. Converter.UseSwitchStatement:=true;
  8375. Add('var vI: longint;');
  8376. Add('begin');
  8377. Add(' case vi of');
  8378. Add(' 1: begin VI:=2; vi:=3; end;');
  8379. Add(' end;');
  8380. ConvertProgram;
  8381. CheckSource('TestCaseOfNoElse_UseSwitch',
  8382. LinesToStr([ // statements
  8383. 'this.vI = 0;'
  8384. ]),
  8385. LinesToStr([ // $mod.$main
  8386. 'switch ($mod.vI) {',
  8387. 'case 1:',
  8388. ' $mod.vI = 2;',
  8389. ' $mod.vI = 3;',
  8390. ' break;',
  8391. '};'
  8392. ]));
  8393. end;
  8394. procedure TTestModule.TestCaseOfRange;
  8395. begin
  8396. StartProgram(false);
  8397. Add('var vI: longint;');
  8398. Add('begin');
  8399. Add(' case vi of');
  8400. Add(' 1..3: vi:=14;');
  8401. Add(' 4,5: vi:=16;');
  8402. Add(' 6..7,9..10: ;');
  8403. Add(' else ;');
  8404. Add(' end;');
  8405. ConvertProgram;
  8406. CheckSource('TestCaseOfRange',
  8407. LinesToStr([ // statements
  8408. 'this.vI = 0;'
  8409. ]),
  8410. LinesToStr([ // $mod.$main
  8411. 'var $tmp = $mod.vI;',
  8412. 'if (($tmp >= 1) && ($tmp <= 3)){',
  8413. ' $mod.vI = 14',
  8414. '} else if (($tmp === 4) || ($tmp === 5)){',
  8415. ' $mod.vI = 16',
  8416. '} else if ((($tmp >= 6) && ($tmp <= 7)) || (($tmp >= 9) && ($tmp <= 10))) ;'
  8417. ]));
  8418. end;
  8419. procedure TTestModule.TestCaseOfString;
  8420. begin
  8421. StartProgram(false);
  8422. Add([
  8423. 'var s,h: string;',
  8424. 'begin',
  8425. ' case s of',
  8426. ' ''foo'': s:=h;',
  8427. ' ''a''..''z'': h:=s;',
  8428. ' ''ў'', ''ё'': ;',
  8429. ' ''Б''..''Я'': ;',
  8430. ' end;',
  8431. '']);
  8432. ConvertProgram;
  8433. CheckSource('TestCaseOfString',
  8434. LinesToStr([ // statements
  8435. 'this.s = "";',
  8436. 'this.h = "";',
  8437. '']),
  8438. LinesToStr([ // $mod.$main
  8439. 'var $tmp = $mod.s;',
  8440. 'if ($tmp === "foo") {',
  8441. ' $mod.s = $mod.h}',
  8442. ' else if (($tmp.length === 1) && ($tmp >= "a") && ($tmp <= "z")) {',
  8443. ' $mod.h = $mod.s}',
  8444. ' else if (($tmp === "ў") || ($tmp === "ё")) {}',
  8445. ' else if (($tmp.length === 1) && ($tmp >= "Б") && ($tmp <= "Я")) ;',
  8446. '']));
  8447. end;
  8448. procedure TTestModule.TestCaseOfChar;
  8449. begin
  8450. StartProgram(false);
  8451. Add([
  8452. 'var s,h: char;',
  8453. 'begin',
  8454. ' case s of',
  8455. ' ''a''..''z'': h:=s;',
  8456. ' ''ä'': ;',
  8457. ' ''ў'', ''ё'': ;',
  8458. ' ''Б''..''Я'': ;',
  8459. ' end;',
  8460. '']);
  8461. ConvertProgram;
  8462. CheckSource('TestCaseOfString',
  8463. LinesToStr([ // statements
  8464. 'this.s = "";',
  8465. 'this.h = "";',
  8466. '']),
  8467. LinesToStr([ // $mod.$main
  8468. 'var $tmp = $mod.s;',
  8469. 'if (($tmp >= "a") && ($tmp <= "z")) {',
  8470. ' $mod.h = $mod.s}',
  8471. ' else if ($tmp === "ä") {}',
  8472. ' else if (($tmp === "ў") || ($tmp === "ё")) {}',
  8473. ' else if (($tmp >= "Б") && ($tmp <= "Я")) ;',
  8474. '']));
  8475. end;
  8476. procedure TTestModule.TestCaseOfExternalClassConst;
  8477. begin
  8478. StartProgram(false);
  8479. Add([
  8480. '{$modeswitch externalclass}',
  8481. 'type',
  8482. ' TBird = class external name ''Bird''',
  8483. ' const e: longint;',
  8484. ' end;',
  8485. 'var vI: longint;',
  8486. 'begin',
  8487. ' case vi of',
  8488. ' 1: vi:=3;',
  8489. ' TBird.e: ;',
  8490. ' end;']);
  8491. ConvertProgram;
  8492. CheckSource('TestCaseOfExternalClassConst',
  8493. LinesToStr([ // statements
  8494. 'this.vI = 0;'
  8495. ]),
  8496. LinesToStr([ // $mod.$main
  8497. 'var $tmp = $mod.vI;',
  8498. 'if ($tmp === 1) {',
  8499. ' $mod.vI = 3}',
  8500. ' else if ($tmp === Bird.e) ;'
  8501. ]));
  8502. end;
  8503. procedure TTestModule.TestDebugger;
  8504. begin
  8505. StartProgram(false);
  8506. Add([
  8507. 'procedure DoIt;',
  8508. 'begin',
  8509. ' deBugger;',
  8510. ' DeBugger();',
  8511. 'end;',
  8512. 'begin',
  8513. ' Debugger;']);
  8514. ConvertProgram;
  8515. CheckSource('TestDebugger',
  8516. LinesToStr([ // statements
  8517. 'this.DoIt = function () {',
  8518. ' debugger;',
  8519. ' debugger;',
  8520. '};',
  8521. '']),
  8522. LinesToStr([ // $mod.$main
  8523. 'debugger;',
  8524. '']));
  8525. end;
  8526. procedure TTestModule.TestArray_Dynamic;
  8527. begin
  8528. StartProgram(false);
  8529. Add([
  8530. 'type',
  8531. ' TArrayInt = array of longint;',
  8532. 'var',
  8533. ' Arr: TArrayInt;',
  8534. ' i: longint;',
  8535. ' b: boolean;',
  8536. 'begin',
  8537. ' SetLength(arr,3);',
  8538. ' arr[0]:=4;',
  8539. ' arr[1]:=length(arr)+arr[0];',
  8540. ' arr[i]:=5;',
  8541. ' arr[arr[i]]:=arr[6];',
  8542. ' i:=low(arr);',
  8543. ' i:=high(arr);',
  8544. ' b:=Assigned(arr);',
  8545. ' Arr:=default(TArrayInt);']);
  8546. ConvertProgram;
  8547. CheckSource('TestArray_Dynamic',
  8548. LinesToStr([ // statements
  8549. 'this.Arr = [];',
  8550. 'this.i = 0;',
  8551. 'this.b = false;'
  8552. ]),
  8553. LinesToStr([ // $mod.$main
  8554. '$mod.Arr = rtl.arraySetLength($mod.Arr,0,3);',
  8555. '$mod.Arr[0] = 4;',
  8556. '$mod.Arr[1] = rtl.length($mod.Arr) + $mod.Arr[0];',
  8557. '$mod.Arr[$mod.i] = 5;',
  8558. '$mod.Arr[$mod.Arr[$mod.i]] = $mod.Arr[6];',
  8559. '$mod.i = 0;',
  8560. '$mod.i = rtl.length($mod.Arr) - 1;',
  8561. '$mod.b = rtl.length($mod.Arr) > 0;',
  8562. '$mod.Arr = [];',
  8563. '']));
  8564. end;
  8565. procedure TTestModule.TestArray_Dynamic_Nil;
  8566. begin
  8567. StartProgram(false);
  8568. Add('type');
  8569. Add(' TArrayInt = array of longint;');
  8570. Add('var');
  8571. Add(' Arr: TArrayInt;');
  8572. Add('procedure DoIt(const i: TArrayInt; j: TArrayInt); begin end;');
  8573. Add('begin');
  8574. Add(' arr:=nil;');
  8575. Add(' if arr=nil then;');
  8576. Add(' if nil=arr then;');
  8577. Add(' if arr<>nil then;');
  8578. Add(' if nil<>arr then;');
  8579. Add(' DoIt(nil,nil);');
  8580. ConvertProgram;
  8581. CheckSource('TestArray_Dynamic',
  8582. LinesToStr([ // statements
  8583. 'this.Arr = [];',
  8584. 'this.DoIt = function(i,j){',
  8585. '};'
  8586. ]),
  8587. LinesToStr([ // $mod.$main
  8588. '$mod.Arr = [];',
  8589. 'if (rtl.length($mod.Arr) === 0) ;',
  8590. 'if (rtl.length($mod.Arr) === 0) ;',
  8591. 'if (rtl.length($mod.Arr) > 0) ;',
  8592. 'if (rtl.length($mod.Arr) > 0) ;',
  8593. '$mod.DoIt([],[]);',
  8594. '']));
  8595. end;
  8596. procedure TTestModule.TestArray_DynMultiDimensional;
  8597. begin
  8598. StartProgram(false);
  8599. Add([
  8600. 'type',
  8601. ' TArrayInt = array of longint;',
  8602. ' TArrayArrayInt = array of TArrayInt;',
  8603. 'var',
  8604. ' Arr: TArrayInt;',
  8605. ' Arr2: TArrayArrayInt;',
  8606. ' i: longint;',
  8607. 'begin',
  8608. ' arr2:=nil;',
  8609. ' if arr2=nil then;',
  8610. ' if nil=arr2 then;',
  8611. ' i:=low(arr2);',
  8612. ' i:=low(arr2[1]);',
  8613. ' i:=high(arr2);',
  8614. ' i:=high(arr2[2]);',
  8615. ' arr2[3]:=arr;',
  8616. ' arr2[4][5]:=i;',
  8617. ' i:=arr2[6][7];',
  8618. ' arr2[8,9]:=i;',
  8619. ' i:=arr2[10,11];',
  8620. ' SetLength(arr2,14);',
  8621. ' SetLength(arr2[15],16);']);
  8622. ConvertProgram;
  8623. CheckSource('TestArray_Dynamic',
  8624. LinesToStr([ // statements
  8625. 'this.Arr = [];',
  8626. 'this.Arr2 = [];',
  8627. 'this.i = 0;'
  8628. ]),
  8629. LinesToStr([ // $mod.$main
  8630. '$mod.Arr2 = [];',
  8631. 'if (rtl.length($mod.Arr2) === 0) ;',
  8632. 'if (rtl.length($mod.Arr2) === 0) ;',
  8633. '$mod.i = 0;',
  8634. '$mod.i = 0;',
  8635. '$mod.i = rtl.length($mod.Arr2) - 1;',
  8636. '$mod.i = rtl.length($mod.Arr2[2]) - 1;',
  8637. '$mod.Arr2[3] = rtl.arrayRef($mod.Arr);',
  8638. '$mod.Arr2[4][5] = $mod.i;',
  8639. '$mod.i = $mod.Arr2[6][7];',
  8640. '$mod.Arr2[8][9] = $mod.i;',
  8641. '$mod.i = $mod.Arr2[10][11];',
  8642. '$mod.Arr2 = rtl.arraySetLength($mod.Arr2, [], 14);',
  8643. '$mod.Arr2[15] = rtl.arraySetLength($mod.Arr2[15], 0, 16);',
  8644. '']));
  8645. end;
  8646. procedure TTestModule.TestArray_DynamicAssign;
  8647. begin
  8648. StartProgram(false);
  8649. Add([
  8650. 'type',
  8651. ' TArrayInt = array of longint;',
  8652. ' TArrayArrayInt = array of TArrayInt;',
  8653. 'procedure Run(a: TArrayInt; const b: TArrayInt; constref c: TArrayInt);',
  8654. 'begin',
  8655. 'end;',
  8656. 'procedure Fly(var a: TArrayInt);',
  8657. 'begin',
  8658. 'end;',
  8659. 'var',
  8660. ' Arr: TArrayInt;',
  8661. ' Arr2: TArrayArrayInt;',
  8662. 'begin',
  8663. ' arr:=nil;',
  8664. ' arr2:=nil;',
  8665. ' arr2[1]:=nil;',
  8666. ' arr2[2]:=arr;',
  8667. ' Run(arr,arr,arr);',
  8668. ' Fly(arr);',
  8669. ' Run(arr2[4],arr2[5],arr2[6]);',
  8670. ' Fly(arr2[7]);',
  8671. '']);
  8672. ConvertProgram;
  8673. CheckSource('TestArray_DynamicAssign',
  8674. LinesToStr([ // statements
  8675. 'this.Run = function (a, b, c) {',
  8676. '};',
  8677. 'this.Fly = function (a) {',
  8678. '};',
  8679. 'this.Arr = [];',
  8680. 'this.Arr2 = [];',
  8681. '']),
  8682. LinesToStr([ // $mod.$main
  8683. '$mod.Arr = [];',
  8684. '$mod.Arr2 = [];',
  8685. '$mod.Arr2[1] = [];',
  8686. '$mod.Arr2[2] = rtl.arrayRef($mod.Arr);',
  8687. '$mod.Run(rtl.arrayRef($mod.Arr), $mod.Arr, $mod.Arr);',
  8688. '$mod.Fly({',
  8689. ' p: $mod,',
  8690. ' get: function () {',
  8691. ' return this.p.Arr;',
  8692. ' },',
  8693. ' set: function (v) {',
  8694. ' this.p.Arr = v;',
  8695. ' }',
  8696. '});',
  8697. '$mod.Run(rtl.arrayRef($mod.Arr2[4]), $mod.Arr2[5], $mod.Arr2[6]);',
  8698. '$mod.Fly({',
  8699. ' a: 7,',
  8700. ' p: $mod.Arr2,',
  8701. ' get: function () {',
  8702. ' return this.p[this.a];',
  8703. ' },',
  8704. ' set: function (v) {',
  8705. ' this.p[this.a] = v;',
  8706. ' }',
  8707. '});',
  8708. '']));
  8709. end;
  8710. procedure TTestModule.TestArray_StaticInt;
  8711. begin
  8712. StartProgram(false);
  8713. Add('type');
  8714. Add(' TArrayInt = array[2..4] of longint;');
  8715. Add('var');
  8716. Add(' Arr: TArrayInt;');
  8717. Add(' Arr2: TArrayInt = (5,6,7);');
  8718. Add(' i: longint;');
  8719. Add(' b: boolean;');
  8720. Add('begin');
  8721. Add(' arr[2]:=4;');
  8722. Add(' arr[3]:=arr[2]+arr[3];');
  8723. Add(' arr[i]:=5;');
  8724. Add(' arr[arr[i]]:=arr[high(arr)];');
  8725. Add(' i:=low(arr);');
  8726. Add(' i:=high(arr);');
  8727. Add(' b:=arr[2]=arr[3];');
  8728. Add(' arr:=default(TArrayInt);');
  8729. ConvertProgram;
  8730. CheckSource('TestArray_StaticInt',
  8731. LinesToStr([ // statements
  8732. 'this.Arr = rtl.arraySetLength(null,0,3);',
  8733. 'this.Arr2 = [5, 6, 7];',
  8734. 'this.i = 0;',
  8735. 'this.b = false;'
  8736. ]),
  8737. LinesToStr([ // $mod.$main
  8738. '$mod.Arr[0] = 4;',
  8739. '$mod.Arr[1] = $mod.Arr[0] + $mod.Arr[1];',
  8740. '$mod.Arr[$mod.i-2] = 5;',
  8741. '$mod.Arr[$mod.Arr[$mod.i-2]-2] = $mod.Arr[2];',
  8742. '$mod.i = 2;',
  8743. '$mod.i = 4;',
  8744. '$mod.b = $mod.Arr[0] === $mod.Arr[1];',
  8745. '$mod.Arr = rtl.arraySetLength(null,0,3);',
  8746. '']));
  8747. end;
  8748. procedure TTestModule.TestArray_StaticBool;
  8749. begin
  8750. StartProgram(false);
  8751. Add('type');
  8752. Add(' TBools = array[boolean] of boolean;');
  8753. Add(' TBool2 = array[true..true] of boolean;');
  8754. Add('var');
  8755. Add(' Arr: TBools;');
  8756. Add(' Arr2: TBool2;');
  8757. Add(' Arr3: TBools = (true,false);');
  8758. Add(' b: boolean;');
  8759. Add('begin');
  8760. Add(' b:=low(arr);');
  8761. Add(' b:=high(arr);');
  8762. Add(' arr[true]:=false;');
  8763. Add(' arr[false]:=arr[b] or arr[true];');
  8764. Add(' arr[b]:=true;');
  8765. Add(' arr[arr[b]]:=arr[high(arr)];');
  8766. Add(' b:=arr[false]=arr[true];');
  8767. Add(' b:=low(arr2);');
  8768. Add(' b:=high(arr2);');
  8769. Add(' arr2[true]:=true;');
  8770. Add(' arr2[true]:=arr2[true] and arr2[b];');
  8771. Add(' arr2[b]:=false;');
  8772. ConvertProgram;
  8773. CheckSource('TestArray_StaticBool',
  8774. LinesToStr([ // statements
  8775. 'this.Arr = rtl.arraySetLength(null,false,2);',
  8776. 'this.Arr2 = rtl.arraySetLength(null,false,1);',
  8777. 'this.Arr3 = [true, false];',
  8778. 'this.b = false;'
  8779. ]),
  8780. LinesToStr([ // $mod.$main
  8781. '$mod.b = false;',
  8782. '$mod.b = true;',
  8783. '$mod.Arr[1] = false;',
  8784. '$mod.Arr[0] = $mod.Arr[+$mod.b] || $mod.Arr[1];',
  8785. '$mod.Arr[+$mod.b] = true;',
  8786. '$mod.Arr[+$mod.Arr[+$mod.b]] = $mod.Arr[1];',
  8787. '$mod.b = $mod.Arr[0] === $mod.Arr[1];',
  8788. '$mod.b = true;',
  8789. '$mod.b = true;',
  8790. '$mod.Arr2[0] = true;',
  8791. '$mod.Arr2[0] = $mod.Arr2[0] && $mod.Arr2[1-$mod.b];',
  8792. '$mod.Arr2[1-$mod.b] = false;',
  8793. '']));
  8794. end;
  8795. procedure TTestModule.TestArray_StaticChar;
  8796. begin
  8797. StartProgram(false);
  8798. Add([
  8799. 'type',
  8800. ' TChars = array[char] of char;',
  8801. ' TChars2 = array[''a''..''z''] of char;',
  8802. 'var',
  8803. ' Arr: TChars;',
  8804. ' Arr2: TChars2;',
  8805. ' Arr3: array[2..4] of char = (''p'',''a'',''s'');',
  8806. ' Arr4: array[11..13] of char = ''pas'';',
  8807. ' Arr5: array[21..22] of char = ''äö'';',
  8808. ' Arr6: array[31..32] of char = ''ä''+''ö'';',
  8809. ' c: char;',
  8810. ' b: boolean;',
  8811. 'begin',
  8812. ' c:=low(arr);',
  8813. ' c:=high(arr);',
  8814. ' arr[''B'']:=''a'';',
  8815. ' arr[''D'']:=arr[c];',
  8816. ' arr[c]:=arr[''d''];',
  8817. ' arr[arr[c]]:=arr[high(arr)];',
  8818. ' b:=arr[low(arr)]=arr[''e''];',
  8819. ' c:=low(arr2);',
  8820. ' c:=high(arr2);',
  8821. ' arr2[''b'']:=''f'';',
  8822. ' arr2[''a'']:=arr2[c];',
  8823. ' arr2[c]:=arr2[''g''];']);
  8824. ConvertProgram;
  8825. CheckSource('TestArray_StaticChar',
  8826. LinesToStr([ // statements
  8827. 'this.Arr = rtl.arraySetLength(null, "", 65536);',
  8828. 'this.Arr2 = rtl.arraySetLength(null, "", 26);',
  8829. 'this.Arr3 = ["p", "a", "s"];',
  8830. 'this.Arr4 = ["p", "a", "s"];',
  8831. 'this.Arr5 = ["ä", "ö"];',
  8832. 'this.Arr6 = ["ä", "ö"];',
  8833. 'this.c = "";',
  8834. 'this.b = false;',
  8835. '']),
  8836. LinesToStr([ // $mod.$main
  8837. '$mod.c = "\x00";',
  8838. '$mod.c = "\uFFFF";',
  8839. '$mod.Arr[66] = "a";',
  8840. '$mod.Arr[68] = $mod.Arr[$mod.c.charCodeAt()];',
  8841. '$mod.Arr[$mod.c.charCodeAt()] = $mod.Arr[100];',
  8842. '$mod.Arr[$mod.Arr[$mod.c.charCodeAt()].charCodeAt()] = $mod.Arr[65535];',
  8843. '$mod.b = $mod.Arr[0] === $mod.Arr[101];',
  8844. '$mod.c = "a";',
  8845. '$mod.c = "z";',
  8846. '$mod.Arr2[1] = "f";',
  8847. '$mod.Arr2[0] = $mod.Arr2[$mod.c.charCodeAt() - 97];',
  8848. '$mod.Arr2[$mod.c.charCodeAt() - 97] = $mod.Arr2[6];',
  8849. '']));
  8850. end;
  8851. procedure TTestModule.TestArray_StaticMultiDim;
  8852. begin
  8853. StartProgram(false);
  8854. Add([
  8855. 'type',
  8856. ' TArrayInt = array[1..3] of longint;',
  8857. ' TArrayArrayInt = array[5..6] of TArrayInt;',
  8858. 'var',
  8859. ' Arr: TArrayInt;',
  8860. ' Arr2: TArrayArrayInt;',
  8861. ' Arr3: array[boolean] of TArrayInt = ((11,12,13),(21,22,23));',
  8862. ' i: longint;',
  8863. 'begin',
  8864. ' i:=low(arr);',
  8865. ' i:=low(arr2);',
  8866. ' i:=low(arr2[5]);',
  8867. ' i:=high(arr);',
  8868. ' i:=high(arr2);',
  8869. ' i:=high(arr2[6]);',
  8870. ' arr2[5]:=arr;',
  8871. ' arr2[6][2]:=i;',
  8872. ' i:=arr2[6][3];',
  8873. ' arr2[6,3]:=i;',
  8874. ' i:=arr2[5,2];',
  8875. ' arr2:=arr2;',// clone multi dim static array
  8876. //' arr3:=arr3;',// clone anonymous multi dim static array
  8877. '']);
  8878. ConvertProgram;
  8879. CheckSource('TestArray_StaticMultiDim',
  8880. LinesToStr([ // statements
  8881. 'this.TArrayArrayInt$clone = function (a) {',
  8882. ' var r = [];',
  8883. ' for (var i = 0; i < 2; i++) r.push(a[i].slice(0));',
  8884. ' return r;',
  8885. '};',
  8886. 'this.Arr = rtl.arraySetLength(null, 0, 3);',
  8887. 'this.Arr2 = rtl.arraySetLength(null, 0, 2, 3);',
  8888. 'this.Arr3 = [[11, 12, 13], [21, 22, 23]];',
  8889. 'this.i = 0;'
  8890. ]),
  8891. LinesToStr([ // $mod.$main
  8892. '$mod.i = 1;',
  8893. '$mod.i = 5;',
  8894. '$mod.i = 1;',
  8895. '$mod.i = 3;',
  8896. '$mod.i = 6;',
  8897. '$mod.i = 3;',
  8898. '$mod.Arr2[0] = $mod.Arr.slice(0);',
  8899. '$mod.Arr2[1][1] = $mod.i;',
  8900. '$mod.i = $mod.Arr2[1][2];',
  8901. '$mod.Arr2[1][2] = $mod.i;',
  8902. '$mod.i = $mod.Arr2[0][1];',
  8903. '$mod.Arr2 = $mod.TArrayArrayInt$clone($mod.Arr2);',
  8904. '']));
  8905. end;
  8906. procedure TTestModule.TestArray_StaticInFunction;
  8907. begin
  8908. StartProgram(false);
  8909. Add([
  8910. 'const TArrayInt = 3;',
  8911. 'const TArrayArrayInt = 4;',
  8912. 'procedure DoIt;',
  8913. 'type',
  8914. ' TArrayInt = array[1..3] of longint;',
  8915. ' TArrayArrayInt = array[5..6] of TArrayInt;',
  8916. 'var',
  8917. ' Arr: TArrayInt;',
  8918. ' Arr2: TArrayArrayInt;',
  8919. ' Arr3: array[boolean] of TArrayInt = ((11,12,13),(21,22,23));',
  8920. ' i: longint;',
  8921. 'begin',
  8922. ' arr2[5]:=arr;',
  8923. ' arr2:=arr2;',// clone multi dim static array
  8924. 'end;',
  8925. 'begin',
  8926. '']);
  8927. ConvertProgram;
  8928. CheckSource('TestArray_StaticInFunction',
  8929. LinesToStr([ // statements
  8930. 'this.TArrayInt = 3;',
  8931. 'this.TArrayArrayInt = 4;',
  8932. 'var TArrayArrayInt$1$clone = function (a) {',
  8933. ' var r = [];',
  8934. ' for (var i = 0; i < 2; i++) r.push(a[i].slice(0));',
  8935. ' return r;',
  8936. '};',
  8937. 'this.DoIt = function () {',
  8938. ' var Arr = rtl.arraySetLength(null, 0, 3);',
  8939. ' var Arr2 = rtl.arraySetLength(null, 0, 2, 3);',
  8940. ' var Arr3 = [[11, 12, 13], [21, 22, 23]];',
  8941. ' var i = 0;',
  8942. ' Arr2[0] = Arr.slice(0);',
  8943. ' Arr2 = TArrayArrayInt$1$clone(Arr2);',
  8944. '};',
  8945. '']),
  8946. LinesToStr([ // $mod.$main
  8947. '']));
  8948. end;
  8949. procedure TTestModule.TestArray_StaticMultiDimEqualNotImplemented;
  8950. begin
  8951. StartProgram(false);
  8952. Add([
  8953. 'type',
  8954. ' TArrayInt = array[1..3,1..2] of longint;',
  8955. 'var',
  8956. ' a,b: TArrayInt;',
  8957. 'begin',
  8958. ' if a=b then ;',
  8959. '']);
  8960. SetExpectedPasResolverError('compare static array is not supported',
  8961. nXIsNotSupported);
  8962. ConvertProgram;
  8963. end;
  8964. procedure TTestModule.TestArrayOfRecord;
  8965. begin
  8966. StartProgram(false);
  8967. Add([
  8968. 'type',
  8969. ' TRec = record',
  8970. ' Int: longint;',
  8971. ' end;',
  8972. ' TArrayRec = array of TRec;',
  8973. 'procedure DoIt(vd: TRec; const vc: TRec; var vv: TRec);',
  8974. 'begin',
  8975. 'end;',
  8976. 'var',
  8977. ' Arr: TArrayRec;',
  8978. ' r: TRec;',
  8979. ' i: longint;',
  8980. 'begin',
  8981. ' SetLength(arr,3);',
  8982. ' arr[0].int:=4;',
  8983. ' arr[1].int:=length(arr)+arr[2].int;',
  8984. ' arr[arr[i].int].int:=arr[5].int;',
  8985. ' arr[7]:=r;',
  8986. ' r:=arr[8];',
  8987. ' i:=low(arr);',
  8988. ' i:=high(arr);',
  8989. ' DoIt(Arr[9],Arr[10],Arr[11]);']);
  8990. ConvertProgram;
  8991. CheckSource('TestArrayOfRecord',
  8992. LinesToStr([ // statements
  8993. 'rtl.recNewT(this, "TRec", function () {',
  8994. ' this.Int = 0;',
  8995. ' this.$eq = function (b) {',
  8996. ' return this.Int === b.Int;',
  8997. ' };',
  8998. ' this.$assign = function (s) {',
  8999. ' this.Int = s.Int;',
  9000. ' return this;',
  9001. ' };',
  9002. '});',
  9003. 'this.DoIt = function (vd, vc, vv) {',
  9004. '};',
  9005. 'this.Arr = [];',
  9006. 'this.r = this.TRec.$new();',
  9007. 'this.i = 0;'
  9008. ]),
  9009. LinesToStr([ // $mod.$main
  9010. '$mod.Arr = rtl.arraySetLength($mod.Arr,$mod.TRec,3);',
  9011. '$mod.Arr[0].Int = 4;',
  9012. '$mod.Arr[1].Int = rtl.length($mod.Arr)+$mod.Arr[2].Int;',
  9013. '$mod.Arr[$mod.Arr[$mod.i].Int].Int = $mod.Arr[5].Int;',
  9014. '$mod.Arr[7].$assign($mod.r);',
  9015. '$mod.r.$assign($mod.Arr[8]);',
  9016. '$mod.i = 0;',
  9017. '$mod.i = rtl.length($mod.Arr)-1;',
  9018. '$mod.DoIt($mod.TRec.$clone($mod.Arr[9]), $mod.Arr[10], $mod.Arr[11]);',
  9019. '']));
  9020. end;
  9021. procedure TTestModule.TestArray_StaticRecord;
  9022. begin
  9023. StartProgram(false);
  9024. Add([
  9025. 'type',
  9026. ' TRec = record',
  9027. ' Int: longint;',
  9028. ' end;',
  9029. ' TArrayRec = array[1..2] of TRec;',
  9030. 'var',
  9031. ' Arr: TArrayRec;',
  9032. 'begin',
  9033. ' arr[1].int:=length(arr)+low(arr)+high(arr);',
  9034. '']);
  9035. ConvertProgram;
  9036. CheckSource('TestArray_StaticRecord',
  9037. LinesToStr([ // statements
  9038. 'rtl.recNewT(this, "TRec", function () {',
  9039. ' this.Int = 0;',
  9040. ' this.$eq = function (b) {',
  9041. ' return this.Int === b.Int;',
  9042. ' };',
  9043. ' this.$assign = function (s) {',
  9044. ' this.Int = s.Int;',
  9045. ' return this;',
  9046. ' };',
  9047. '});',
  9048. 'this.TArrayRec$clone = function (a) {',
  9049. ' var r = [];',
  9050. ' for (var i = 0; i < 2; i++) r.push($mod.TRec.$clone(a[i]));',
  9051. ' return r;',
  9052. '};',
  9053. 'this.Arr = rtl.arraySetLength(null, this.TRec, 2);',
  9054. '']),
  9055. LinesToStr([ // $mod.$main
  9056. '$mod.Arr[0].Int = 2 + 1 + 2;']));
  9057. end;
  9058. procedure TTestModule.TestArrayOfSet;
  9059. begin
  9060. StartProgram(false);
  9061. Add([
  9062. 'type',
  9063. ' TFlag = (big,small);',
  9064. ' TSetOfFlag = set of tflag;',
  9065. ' TArrayFlag = array of TSetOfFlag;',
  9066. 'procedure DoIt(const a: Tarrayflag);',
  9067. 'begin',
  9068. 'end;',
  9069. 'var',
  9070. ' f: TFlag;',
  9071. ' s: TSetOfFlag;',
  9072. ' Arr: TArrayFlag;',
  9073. ' i: longint;',
  9074. 'begin',
  9075. ' SetLength(arr,3);',
  9076. ' arr[0]:=s;',
  9077. ' arr[1]:=[big];',
  9078. ' arr[2]:=[big]+s;',
  9079. ' arr[3]:=s+[big];',
  9080. ' arr[4]:=arr[5];',
  9081. ' s:=arr[6];',
  9082. ' i:=low(arr);',
  9083. ' i:=high(arr);',
  9084. ' DoIt(arr);',
  9085. ' DoIt([s]);',
  9086. ' DoIt([[],s]);',
  9087. ' DoIt([s,[]]);',
  9088. '']);
  9089. ConvertProgram;
  9090. CheckSource('TestArrayOfSet',
  9091. LinesToStr([ // statements
  9092. 'this.TFlag = {',
  9093. ' "0": "big",',
  9094. ' big: 0,',
  9095. ' "1": "small",',
  9096. ' small: 1',
  9097. '};',
  9098. 'this.DoIt = function (a) {',
  9099. '};',
  9100. 'this.f = 0;',
  9101. 'this.s = {};',
  9102. 'this.Arr = [];',
  9103. 'this.i = 0;',
  9104. '']),
  9105. LinesToStr([ // $mod.$main
  9106. '$mod.Arr = rtl.arraySetLength($mod.Arr, {}, 3);',
  9107. '$mod.Arr[0] = rtl.refSet($mod.s);',
  9108. '$mod.Arr[1] = rtl.createSet($mod.TFlag.big);',
  9109. '$mod.Arr[2] = rtl.unionSet(rtl.createSet($mod.TFlag.big), $mod.s);',
  9110. '$mod.Arr[3] = rtl.unionSet($mod.s, rtl.createSet($mod.TFlag.big));',
  9111. '$mod.Arr[4] = rtl.refSet($mod.Arr[5]);',
  9112. '$mod.s = rtl.refSet($mod.Arr[6]);',
  9113. '$mod.i = 0;',
  9114. '$mod.i = rtl.length($mod.Arr) - 1;',
  9115. '$mod.DoIt($mod.Arr);',
  9116. '$mod.DoIt([rtl.refSet($mod.s)]);',
  9117. '$mod.DoIt([{}, rtl.refSet($mod.s)]);',
  9118. '$mod.DoIt([rtl.refSet($mod.s), {}]);',
  9119. '']));
  9120. end;
  9121. procedure TTestModule.TestArray_DynAsParam;
  9122. begin
  9123. StartProgram(false);
  9124. Add([
  9125. 'type integer = longint;',
  9126. 'type TArrInt = array of integer;',
  9127. 'procedure DoIt(vG: TArrInt; const vH: TArrInt; var vI: TArrInt);',
  9128. 'var vJ: TArrInt;',
  9129. 'begin',
  9130. ' vg:=vg;',
  9131. ' vj:=vh;',
  9132. ' vi:=vi;',
  9133. ' doit(vg,vg,vg);',
  9134. ' doit(vh,vh,vj);',
  9135. ' doit(vi,vi,vi);',
  9136. ' doit(vj,vj,vj);',
  9137. 'end;',
  9138. 'var i: TArrInt;',
  9139. 'begin',
  9140. ' doit(i,i,i);']);
  9141. ConvertProgram;
  9142. CheckSource('TestArray_DynAsParams',
  9143. LinesToStr([ // statements
  9144. 'this.DoIt = function (vG,vH,vI) {',
  9145. ' var vJ = [];',
  9146. ' vG = rtl.arrayRef(vG);',
  9147. ' vJ = rtl.arrayRef(vH);',
  9148. ' vI.set(rtl.arrayRef(vI.get()));',
  9149. ' $mod.DoIt(rtl.arrayRef(vG), vG, {',
  9150. ' get: function () {',
  9151. ' return vG;',
  9152. ' },',
  9153. ' set: function (v) {',
  9154. ' vG = v;',
  9155. ' }',
  9156. ' });',
  9157. ' $mod.DoIt(rtl.arrayRef(vH), vH, {',
  9158. ' get: function () {',
  9159. ' return vJ;',
  9160. ' },',
  9161. ' set: function (v) {',
  9162. ' vJ = v;',
  9163. ' }',
  9164. ' });',
  9165. ' $mod.DoIt(rtl.arrayRef(vI.get()), vI.get(), vI);',
  9166. ' $mod.DoIt(rtl.arrayRef(vJ), vJ, {',
  9167. ' get: function () {',
  9168. ' return vJ;',
  9169. ' },',
  9170. ' set: function (v) {',
  9171. ' vJ = v;',
  9172. ' }',
  9173. ' });',
  9174. '};',
  9175. 'this.i = [];'
  9176. ]),
  9177. LinesToStr([
  9178. '$mod.DoIt(rtl.arrayRef($mod.i),$mod.i,{',
  9179. ' p: $mod,',
  9180. ' get: function () {',
  9181. ' return this.p.i;',
  9182. ' },',
  9183. ' set: function (v) {',
  9184. ' this.p.i = v;',
  9185. ' }',
  9186. '});'
  9187. ]));
  9188. end;
  9189. procedure TTestModule.TestArray_StaticAsParam;
  9190. begin
  9191. StartProgram(false);
  9192. Add([
  9193. 'type integer = longint;',
  9194. 'type TArrInt = array[1..2] of integer;',
  9195. 'procedure DoIt(vG: TArrInt; const vH: TArrInt; var vI: TArrInt);',
  9196. 'var vJ: TArrInt;',
  9197. 'begin',
  9198. ' vg:=vg;',
  9199. ' vj:=vh;',
  9200. ' vi:=vi;',
  9201. ' doit(vg,vg,vg);',
  9202. ' doit(vh,vh,vj);',
  9203. ' doit(vi,vi,vi);',
  9204. ' doit(vj,vj,vj);',
  9205. 'end;',
  9206. 'var i: TArrInt;',
  9207. 'begin',
  9208. ' doit(i,i,i);']);
  9209. ConvertProgram;
  9210. CheckSource('TestArray_StaticAsParams',
  9211. LinesToStr([ // statements
  9212. 'this.DoIt = function (vG,vH,vI) {',
  9213. ' var vJ = rtl.arraySetLength(null, 0, 2);',
  9214. ' vG = vG.slice(0);',
  9215. ' vJ = vH.slice(0);',
  9216. ' vI.set(vI.get().slice(0));',
  9217. ' $mod.DoIt(vG.slice(0), vG, {',
  9218. ' get: function () {',
  9219. ' return vG;',
  9220. ' },',
  9221. ' set: function (v) {',
  9222. ' vG = v;',
  9223. ' }',
  9224. ' });',
  9225. ' $mod.DoIt(vH.slice(0), vH, {',
  9226. ' get: function () {',
  9227. ' return vJ;',
  9228. ' },',
  9229. ' set: function (v) {',
  9230. ' vJ = v;',
  9231. ' }',
  9232. ' });',
  9233. ' $mod.DoIt(vI.get().slice(0), vI.get(), vI);',
  9234. ' $mod.DoIt(vJ.slice(0), vJ, {',
  9235. ' get: function () {',
  9236. ' return vJ;',
  9237. ' },',
  9238. ' set: function (v) {',
  9239. ' vJ = v;',
  9240. ' }',
  9241. ' });',
  9242. '};',
  9243. 'this.i = rtl.arraySetLength(null, 0, 2);'
  9244. ]),
  9245. LinesToStr([
  9246. '$mod.DoIt($mod.i.slice(0),$mod.i,{',
  9247. ' p: $mod,',
  9248. ' get: function () {',
  9249. ' return this.p.i;',
  9250. ' },',
  9251. ' set: function (v) {',
  9252. ' this.p.i = v;',
  9253. ' }',
  9254. '});'
  9255. ]));
  9256. end;
  9257. procedure TTestModule.TestArrayElement_AsParams;
  9258. begin
  9259. StartProgram(false);
  9260. Add('type integer = longint;');
  9261. Add('type TArrayInt = array of integer;');
  9262. Add('procedure DoIt(vG: Integer; const vH: Integer; var vI: Integer);');
  9263. Add('var vJ: tarrayint;');
  9264. Add('begin');
  9265. Add(' vi:=vi;');
  9266. Add(' doit(vi,vi,vi);');
  9267. Add(' doit(vj[1+1],vj[1+2],vj[1+3]);');
  9268. Add('end;');
  9269. Add('var a: TArrayInt;');
  9270. Add('begin');
  9271. Add(' doit(a[1+4],a[1+5],a[1+6]);');
  9272. ConvertProgram;
  9273. CheckSource('TestArrayElement_AsParams',
  9274. LinesToStr([ // statements
  9275. 'this.DoIt = function (vG,vH,vI) {',
  9276. ' var vJ = [];',
  9277. ' vI.set(vI.get());',
  9278. ' $mod.DoIt(vI.get(), vI.get(), vI);',
  9279. ' $mod.DoIt(vJ[1+1], vJ[1+2], {',
  9280. ' a:1+3,',
  9281. ' p:vJ,',
  9282. ' get: function () {',
  9283. ' return this.p[this.a];',
  9284. ' },',
  9285. ' set: function (v) {',
  9286. ' this.p[this.a] = v;',
  9287. ' }',
  9288. ' });',
  9289. '};',
  9290. 'this.a = [];'
  9291. ]),
  9292. LinesToStr([
  9293. '$mod.DoIt($mod.a[1+4],$mod.a[1+5],{',
  9294. ' a: 1+6,',
  9295. ' p: $mod.a,',
  9296. ' get: function () {',
  9297. ' return this.p[this.a];',
  9298. ' },',
  9299. ' set: function (v) {',
  9300. ' this.p[this.a] = v;',
  9301. ' }',
  9302. '});'
  9303. ]));
  9304. end;
  9305. procedure TTestModule.TestArrayElementFromFuncResult_AsParams;
  9306. begin
  9307. StartProgram(false);
  9308. Add('type Integer = longint;');
  9309. Add('type TArrayInt = array of integer;');
  9310. Add('function GetArr(vB: integer = 0): tarrayint;');
  9311. Add('begin');
  9312. Add('end;');
  9313. Add('procedure DoIt(vG: integer; const vH: integer; var vI: integer);');
  9314. Add('begin');
  9315. Add('end;');
  9316. Add('begin');
  9317. Add(' doit(getarr[1+1],getarr[1+2],getarr[1+3]);');
  9318. Add(' doit(getarr()[2+1],getarr()[2+2],getarr()[2+3]);');
  9319. Add(' doit(getarr(7)[3+1],getarr(8)[3+2],getarr(9)[3+3]);');
  9320. ConvertProgram;
  9321. CheckSource('TestArrayElementFromFuncResult_AsParams',
  9322. LinesToStr([ // statements
  9323. 'this.GetArr = function (vB) {',
  9324. ' var Result = [];',
  9325. ' return Result;',
  9326. '};',
  9327. 'this.DoIt = function (vG,vH,vI) {',
  9328. '};'
  9329. ]),
  9330. LinesToStr([
  9331. '$mod.DoIt($mod.GetArr(0)[1+1],$mod.GetArr(0)[1+2],{',
  9332. ' a: 1+3,',
  9333. ' p: $mod.GetArr(0),',
  9334. ' get: function () {',
  9335. ' return this.p[this.a];',
  9336. ' },',
  9337. ' set: function (v) {',
  9338. ' this.p[this.a] = v;',
  9339. ' }',
  9340. '});',
  9341. '$mod.DoIt($mod.GetArr(0)[2+1],$mod.GetArr(0)[2+2],{',
  9342. ' a: 2+3,',
  9343. ' p: $mod.GetArr(0),',
  9344. ' get: function () {',
  9345. ' return this.p[this.a];',
  9346. ' },',
  9347. ' set: function (v) {',
  9348. ' this.p[this.a] = v;',
  9349. ' }',
  9350. '});',
  9351. '$mod.DoIt($mod.GetArr(7)[3+1],$mod.GetArr(8)[3+2],{',
  9352. ' a: 3+3,',
  9353. ' p: $mod.GetArr(9),',
  9354. ' get: function () {',
  9355. ' return this.p[this.a];',
  9356. ' },',
  9357. ' set: function (v) {',
  9358. ' this.p[this.a] = v;',
  9359. ' }',
  9360. '});',
  9361. '']));
  9362. end;
  9363. procedure TTestModule.TestArrayEnumTypeRange;
  9364. begin
  9365. StartProgram(false);
  9366. Add([
  9367. 'type',
  9368. ' TEnum = (red,blue);',
  9369. ' TEnumArray = array[TEnum] of longint;',
  9370. 'var',
  9371. ' e: TEnum;',
  9372. ' i: longint;',
  9373. ' a: TEnumArray;',
  9374. ' numbers: TEnumArray = (1,2);',
  9375. ' names: array[TEnum] of string = (''red'',''blue'');',
  9376. 'begin',
  9377. ' e:=low(a);',
  9378. ' e:=high(a);',
  9379. ' i:=a[red];',
  9380. ' a[e]:=a[e];']);
  9381. ConvertProgram;
  9382. CheckSource('TestArrayEnumTypeRange',
  9383. LinesToStr([ // statements
  9384. ' this.TEnum = {',
  9385. ' "0": "red",',
  9386. ' red: 0,',
  9387. ' "1": "blue",',
  9388. ' blue: 1',
  9389. '};',
  9390. 'this.e = 0;',
  9391. 'this.i = 0;',
  9392. 'this.a = rtl.arraySetLength(null,0,2);',
  9393. 'this.numbers = [1, 2];',
  9394. 'this.names = ["red", "blue"];',
  9395. '']),
  9396. LinesToStr([ // $mod.$main
  9397. '$mod.e = $mod.TEnum.red;',
  9398. '$mod.e = $mod.TEnum.blue;',
  9399. '$mod.i = $mod.a[$mod.TEnum.red];',
  9400. '$mod.a[$mod.e] = $mod.a[$mod.e];',
  9401. '']));
  9402. end;
  9403. procedure TTestModule.TestArray_SetLengthOutArg;
  9404. begin
  9405. StartProgram(false);
  9406. Add([
  9407. 'type TArrInt = array of longint;',
  9408. 'procedure DoIt(out a: TArrInt);',
  9409. 'begin',
  9410. ' SetLength(a,2);',
  9411. 'end;',
  9412. 'begin',
  9413. '']);
  9414. ConvertProgram;
  9415. CheckSource('TestArray_SetLengthOutArg',
  9416. LinesToStr([ // statements
  9417. 'this.DoIt = function (a) {',
  9418. ' a.set(rtl.arraySetLength(a.get(), 0, 2));',
  9419. '};',
  9420. '']),
  9421. LinesToStr([
  9422. '']));
  9423. end;
  9424. procedure TTestModule.TestArray_SetLengthProperty;
  9425. begin
  9426. StartProgram(false);
  9427. Add('type');
  9428. Add(' TArrInt = array of longint;');
  9429. Add(' TObject = class');
  9430. Add(' function GetColors: TArrInt; external name ''GetColors'';');
  9431. Add(' procedure SetColors(const Value: TArrInt); external name ''SetColors'';');
  9432. Add(' property Colors: TArrInt read GetColors write SetColors;');
  9433. Add(' end;');
  9434. Add('var Obj: TObject;');
  9435. Add('begin');
  9436. Add(' SetLength(Obj.Colors,2);');
  9437. ConvertProgram;
  9438. CheckSource('TestArray_SetLengthProperty',
  9439. LinesToStr([ // statements
  9440. 'rtl.createClass(this, "TObject", null, function () {',
  9441. ' this.$init = function () {',
  9442. ' };',
  9443. ' this.$final = function () {',
  9444. ' };',
  9445. '});',
  9446. 'this.Obj = null;',
  9447. '']),
  9448. LinesToStr([
  9449. '$mod.Obj.SetColors(rtl.arraySetLength($mod.Obj.GetColors(), 0, 2));',
  9450. '']));
  9451. end;
  9452. procedure TTestModule.TestArray_SetLengthMultiDim;
  9453. begin
  9454. StartProgram(false);
  9455. Add([
  9456. 'type',
  9457. ' TArrArrInt = array of array of longint;',
  9458. ' TArrStaInt = array of array[1..2] of longint;',
  9459. 'var',
  9460. ' a: TArrArrInt;',
  9461. ' b: TArrStaInt;',
  9462. 'begin',
  9463. ' SetLength(a,2);',
  9464. ' SetLength(a,3,4);',
  9465. ' SetLength(b,5);',
  9466. '']);
  9467. ConvertProgram;
  9468. CheckSource('TestArray_SetLengthMultiDim',
  9469. LinesToStr([ // statements
  9470. 'this.a = [];',
  9471. 'this.b = [];',
  9472. '']),
  9473. LinesToStr([
  9474. '$mod.a = rtl.arraySetLength($mod.a, [], 2);',
  9475. '$mod.a = rtl.arraySetLength($mod.a, 0, 3, 4);',
  9476. '$mod.b = rtl.arraySetLength($mod.b, 0, 5, "s", 2);',
  9477. '']));
  9478. end;
  9479. procedure TTestModule.TestArray_SetLengthDynOfStatic;
  9480. begin
  9481. StartProgram(false);
  9482. Add([
  9483. 'type',
  9484. ' TStaArr1 = array[1..3] of boolean;',
  9485. //' TStaArr2 = array[5..6] of TStaArr1;',
  9486. ' TDynArr1StaArr1 = array of TStaArr1;',
  9487. //' TDynArr1StaArr2 = array of TStaArr2;',
  9488. ' TDynArr2StaArr1 = array of TDynArr1StaArr1;',
  9489. //' TDynArr2StaArr2 = array of TDynArr1StaArr2;',
  9490. 'var',
  9491. ' DynArr1StaArr1: TDynArr1StaArr1;',
  9492. //' DynArr1StaArr2: TDynArr1StaArr1;',
  9493. ' DynArr2StaArr1: TDynArr2StaArr1;',
  9494. //' DynArr2StaArr2: TDynArr2StaArr2;',
  9495. 'begin',
  9496. ' SetLength(DynArr1StaArr1,11);',
  9497. ' SetLength(DynArr2StaArr1,12);',
  9498. ' SetLength(DynArr2StaArr1[13],14);',
  9499. ' SetLength(DynArr2StaArr1,15,16);',
  9500. //' SetLength(DynArr1StaArr2,21);',
  9501. //' SetLength(DynArr2StaArr2,22);',
  9502. //' SetLength(DynArr2StaArr2[23],24);',
  9503. //' SetLength(DynArr2StaArr2,25,26);',
  9504. '']);
  9505. ConvertProgram;
  9506. CheckSource('TestArray_DynOfStatic',
  9507. LinesToStr([ // statements
  9508. 'this.DynArr1StaArr1 = [];',
  9509. 'this.DynArr2StaArr1 = [];',
  9510. '']),
  9511. LinesToStr([ // $mod.$main
  9512. '$mod.DynArr1StaArr1 = rtl.arraySetLength($mod.DynArr1StaArr1, false, 11, "s", 3);',
  9513. '$mod.DynArr2StaArr1 = rtl.arraySetLength($mod.DynArr2StaArr1, [], 12);',
  9514. '$mod.DynArr2StaArr1[13] = rtl.arraySetLength($mod.DynArr2StaArr1[13], false, 14, "s", 3);',
  9515. '$mod.DynArr2StaArr1 = rtl.arraySetLength(',
  9516. ' $mod.DynArr2StaArr1,',
  9517. ' false,',
  9518. ' 15,',
  9519. ' 16,',
  9520. ' "s",',
  9521. ' 3',
  9522. ');',
  9523. '']));
  9524. end;
  9525. procedure TTestModule.TestArray_OpenArrayOfString;
  9526. begin
  9527. StartProgram(false);
  9528. Add('procedure DoIt(const a: array of String);');
  9529. Add('var');
  9530. Add(' i: longint;');
  9531. Add(' s: string;');
  9532. Add('begin');
  9533. Add(' for i:=low(a) to high(a) do s:=a[length(a)-i-1];');
  9534. Add('end;');
  9535. Add('var s: string;');
  9536. Add('begin');
  9537. Add(' DoIt([]);');
  9538. Add(' DoIt([s,''foo'','''',s+s]);');
  9539. ConvertProgram;
  9540. CheckSource('TestArray_OpenArrayOfString',
  9541. LinesToStr([ // statements
  9542. 'this.DoIt = function (a) {',
  9543. ' var i = 0;',
  9544. ' var s = "";',
  9545. ' for (var $l = 0, $end = rtl.length(a) - 1; $l <= $end; $l++) {',
  9546. ' i = $l;',
  9547. ' s = a[rtl.length(a) - i - 1];',
  9548. ' };',
  9549. '};',
  9550. 'this.s = "";',
  9551. '']),
  9552. LinesToStr([
  9553. '$mod.DoIt([]);',
  9554. '$mod.DoIt([$mod.s, "foo", "", $mod.s + $mod.s]);',
  9555. '']));
  9556. end;
  9557. procedure TTestModule.TestArray_ArrayOfCharAssignString;
  9558. begin
  9559. StartProgram(false);
  9560. Add([
  9561. 'type TArr = array of char;',
  9562. 'var',
  9563. ' c: char;',
  9564. ' s: string;',
  9565. ' a: TArr;',
  9566. 'procedure Run(const a: array of char);',
  9567. 'begin',
  9568. ' Run(c);',
  9569. ' Run(s);',
  9570. 'end;',
  9571. 'begin',
  9572. ' a:=c;',
  9573. ' a:=s;',
  9574. ' a:=#13;',
  9575. ' a:=''Foo'';',
  9576. ' Run(c);',
  9577. ' Run(s);',
  9578. '']);
  9579. ConvertProgram;
  9580. CheckSource('TestArray_ArrayOfCharAssignString',
  9581. LinesToStr([ // statements
  9582. 'this.c = "";',
  9583. 'this.s = "";',
  9584. 'this.a = [];',
  9585. 'this.Run = function (a) {',
  9586. ' $mod.Run($mod.c.split(""));',
  9587. ' $mod.Run($mod.s.split(""));',
  9588. '};',
  9589. '']),
  9590. LinesToStr([
  9591. '$mod.a = $mod.c.split("");',
  9592. '$mod.a = $mod.s.split("");',
  9593. '$mod.a = "\r".split("");',
  9594. '$mod.a = "Foo".split("");',
  9595. '$mod.Run($mod.c.split(""));',
  9596. '$mod.Run($mod.s.split(""));',
  9597. '']));
  9598. end;
  9599. procedure TTestModule.TestArray_ConstRef;
  9600. begin
  9601. StartProgram(false);
  9602. Add([
  9603. 'type TArr = array of word;',
  9604. 'procedure Run(constref a: TArr);',
  9605. 'begin',
  9606. 'end;',
  9607. 'procedure Fly(a: TArr; var b: TArr; out c: TArr; const d: TArr; constref e: TArr);',
  9608. 'var l: TArr;',
  9609. 'begin',
  9610. ' Run(l);',
  9611. ' Run(a);',
  9612. ' Run(b);',
  9613. ' Run(c);',
  9614. ' Run(d);',
  9615. ' Run(e);',
  9616. 'end;',
  9617. 'begin',
  9618. '']);
  9619. ConvertProgram;
  9620. CheckResolverUnexpectedHints();
  9621. CheckSource('TestArray_ConstRef',
  9622. LinesToStr([ // statements
  9623. 'this.Run = function (a) {',
  9624. '};',
  9625. 'this.Fly = function (a, b, c, d, e) {',
  9626. ' var l = [];',
  9627. ' $mod.Run(l);',
  9628. ' $mod.Run(a);',
  9629. ' $mod.Run(b.get());',
  9630. ' $mod.Run(c.get());',
  9631. ' $mod.Run(d);',
  9632. ' $mod.Run(e);',
  9633. '};',
  9634. '']),
  9635. LinesToStr([
  9636. '']));
  9637. end;
  9638. procedure TTestModule.TestArray_Concat;
  9639. begin
  9640. StartProgram(false);
  9641. Add([
  9642. 'type',
  9643. ' integer = longint;',
  9644. ' TFlag = (big,small);',
  9645. ' TFlags = set of TFlag;',
  9646. ' TRec = record',
  9647. ' i: integer;',
  9648. ' end;',
  9649. ' TArrInt = array of integer;',
  9650. ' TArrRec = array of TRec;',
  9651. ' TArrFlag = array of TFlag;',
  9652. ' TArrSet = array of TFlags;',
  9653. ' TArrJSValue = array of jsvalue;',
  9654. 'var',
  9655. ' ArrInt: tarrint;',
  9656. ' ArrRec: tarrrec;',
  9657. ' ArrFlag: tarrflag;',
  9658. ' ArrSet: tarrset;',
  9659. ' ArrJSValue: tarrjsvalue;',
  9660. 'begin',
  9661. ' arrint:=concat(arrint);',
  9662. ' arrint:=concat(arrint,arrint);',
  9663. ' arrint:=concat(arrint,arrint,arrint);',
  9664. ' arrrec:=concat(arrrec);',
  9665. ' arrrec:=concat(arrrec,arrrec);',
  9666. ' arrrec:=concat(arrrec,arrrec,arrrec);',
  9667. ' arrset:=concat(arrset);',
  9668. ' arrset:=concat(arrset,arrset);',
  9669. ' arrset:=concat(arrset,arrset,arrset);',
  9670. ' arrjsvalue:=concat(arrjsvalue);',
  9671. ' arrjsvalue:=concat(arrjsvalue,arrjsvalue);',
  9672. ' arrjsvalue:=concat(arrjsvalue,arrjsvalue,arrjsvalue);',
  9673. ' arrint:=concat([1],arrint);',
  9674. ' arrflag:=concat([big]);',
  9675. ' arrflag:=concat([big],arrflag);',
  9676. ' arrflag:=concat(arrflag,[small]);',
  9677. '']);
  9678. ConvertProgram;
  9679. CheckSource('TestArray_Concat',
  9680. LinesToStr([ // statements
  9681. 'this.TFlag = {',
  9682. ' "0": "big",',
  9683. ' big: 0,',
  9684. ' "1": "small",',
  9685. ' small: 1',
  9686. '};',
  9687. 'rtl.recNewT(this, "TRec", function () {',
  9688. ' this.i = 0;',
  9689. ' this.$eq = function (b) {',
  9690. ' return this.i === b.i;',
  9691. ' };',
  9692. ' this.$assign = function (s) {',
  9693. ' this.i = s.i;',
  9694. ' return this;',
  9695. ' };',
  9696. '});',
  9697. 'this.ArrInt = [];',
  9698. 'this.ArrRec = [];',
  9699. 'this.ArrFlag = [];',
  9700. 'this.ArrSet = [];',
  9701. 'this.ArrJSValue = [];',
  9702. '']),
  9703. LinesToStr([ // $mod.$main
  9704. '$mod.ArrInt = rtl.arrayRef($mod.ArrInt);',
  9705. '$mod.ArrInt = rtl.arrayConcatN($mod.ArrInt, $mod.ArrInt);',
  9706. '$mod.ArrInt = rtl.arrayConcatN($mod.ArrInt, $mod.ArrInt, $mod.ArrInt);',
  9707. '$mod.ArrRec = rtl.arrayRef($mod.ArrRec);',
  9708. '$mod.ArrRec = rtl.arrayConcat($mod.TRec, $mod.ArrRec, $mod.ArrRec);',
  9709. '$mod.ArrRec = rtl.arrayConcat($mod.TRec, $mod.ArrRec, $mod.ArrRec, $mod.ArrRec);',
  9710. '$mod.ArrSet = rtl.arrayRef($mod.ArrSet);',
  9711. '$mod.ArrSet = rtl.arrayConcat("refSet", $mod.ArrSet, $mod.ArrSet);',
  9712. '$mod.ArrSet = rtl.arrayConcat("refSet", $mod.ArrSet, $mod.ArrSet, $mod.ArrSet);',
  9713. '$mod.ArrJSValue = rtl.arrayRef($mod.ArrJSValue);',
  9714. '$mod.ArrJSValue = rtl.arrayConcatN($mod.ArrJSValue, $mod.ArrJSValue);',
  9715. '$mod.ArrJSValue = rtl.arrayConcatN($mod.ArrJSValue, $mod.ArrJSValue, $mod.ArrJSValue);',
  9716. '$mod.ArrInt = rtl.arrayConcatN([1], $mod.ArrInt);',
  9717. '$mod.ArrFlag = [$mod.TFlag.big];',
  9718. '$mod.ArrFlag = rtl.arrayConcatN([$mod.TFlag.big], $mod.ArrFlag);',
  9719. '$mod.ArrFlag = rtl.arrayConcatN($mod.ArrFlag, [$mod.TFlag.small]);',
  9720. '']));
  9721. end;
  9722. procedure TTestModule.TestArray_Copy;
  9723. begin
  9724. StartProgram(false);
  9725. Add([
  9726. 'type',
  9727. ' integer = longint;',
  9728. ' TFlag = (big,small);',
  9729. ' TFlags = set of TFlag;',
  9730. ' TRec = record',
  9731. ' i: integer;',
  9732. ' end;',
  9733. ' TArrInt = array of integer;',
  9734. ' TArrRec = array of TRec;',
  9735. ' TArrSet = array of TFlags;',
  9736. ' TArrJSValue = array of jsvalue;',
  9737. 'var',
  9738. ' ArrInt: tarrint;',
  9739. ' ArrRec: tarrrec;',
  9740. ' ArrSet: tarrset;',
  9741. ' ArrJSValue: tarrjsvalue;',
  9742. 'begin',
  9743. ' arrint:=copy(arrint);',
  9744. ' arrint:=copy(arrint,2);',
  9745. ' arrint:=copy(arrint,3,4);',
  9746. ' arrint:=copy([1,1],1,2);',
  9747. ' arrrec:=copy(arrrec);',
  9748. ' arrrec:=copy(arrrec,5);',
  9749. ' arrrec:=copy(arrrec,6,7);',
  9750. ' arrset:=copy(arrset);',
  9751. ' arrset:=copy(arrset,8);',
  9752. ' arrset:=copy(arrset,9,10);',
  9753. ' arrjsvalue:=copy(arrjsvalue);',
  9754. ' arrjsvalue:=copy(arrjsvalue,11);',
  9755. ' arrjsvalue:=copy(arrjsvalue,12,13);',
  9756. ' ']);
  9757. ConvertProgram;
  9758. CheckSource('TestArray_Copy',
  9759. LinesToStr([ // statements
  9760. 'this.TFlag = {',
  9761. ' "0": "big",',
  9762. ' big: 0,',
  9763. ' "1": "small",',
  9764. ' small: 1',
  9765. '};',
  9766. 'rtl.recNewT(this, "TRec", function () {',
  9767. ' this.i = 0;',
  9768. ' this.$eq = function (b) {',
  9769. ' return this.i === b.i;',
  9770. ' };',
  9771. ' this.$assign = function (s) {',
  9772. ' this.i = s.i;',
  9773. ' return this;',
  9774. ' };',
  9775. '});',
  9776. 'this.ArrInt = [];',
  9777. 'this.ArrRec = [];',
  9778. 'this.ArrSet = [];',
  9779. 'this.ArrJSValue = [];',
  9780. '']),
  9781. LinesToStr([ // $mod.$main
  9782. '$mod.ArrInt = rtl.arrayCopy(0, $mod.ArrInt, 0);',
  9783. '$mod.ArrInt = rtl.arrayCopy(0, $mod.ArrInt, 2);',
  9784. '$mod.ArrInt = rtl.arrayCopy(0, $mod.ArrInt, 3, 4);',
  9785. '$mod.ArrInt = rtl.arrayCopy(0, [1, 1], 1, 2);',
  9786. '$mod.ArrRec = rtl.arrayCopy($mod.TRec, $mod.ArrRec, 0);',
  9787. '$mod.ArrRec = rtl.arrayCopy($mod.TRec, $mod.ArrRec, 5);',
  9788. '$mod.ArrRec = rtl.arrayCopy($mod.TRec, $mod.ArrRec, 6, 7);',
  9789. '$mod.ArrSet = rtl.arrayCopy("refSet", $mod.ArrSet, 0);',
  9790. '$mod.ArrSet = rtl.arrayCopy("refSet", $mod.ArrSet, 8);',
  9791. '$mod.ArrSet = rtl.arrayCopy("refSet", $mod.ArrSet, 9, 10);',
  9792. '$mod.ArrJSValue = rtl.arrayCopy(0, $mod.ArrJSValue, 0);',
  9793. '$mod.ArrJSValue = rtl.arrayCopy(0, $mod.ArrJSValue, 11);',
  9794. '$mod.ArrJSValue = rtl.arrayCopy(0, $mod.ArrJSValue, 12, 13);',
  9795. '']));
  9796. end;
  9797. procedure TTestModule.TestArray_InsertDelete;
  9798. begin
  9799. StartProgram(false);
  9800. Add([
  9801. 'type',
  9802. ' integer = longint;',
  9803. ' TFlag = (big,small);',
  9804. ' TFlags = set of TFlag;',
  9805. ' TRec = record',
  9806. ' i: integer;',
  9807. ' end;',
  9808. ' TArrInt = array of integer;',
  9809. ' TArrRec = array of TRec;',
  9810. ' TArrSet = array of TFlags;',
  9811. ' TArrJSValue = array of jsvalue;',
  9812. ' TArrArrInt = array of TArrInt;',
  9813. 'var',
  9814. ' ArrInt: tarrint;',
  9815. ' ArrRec: tarrrec;',
  9816. ' ArrSet: tarrset;',
  9817. ' ArrJSValue: tarrjsvalue;',
  9818. ' ArrArrInt: TArrArrInt;',
  9819. 'begin',
  9820. ' Insert(1,arrint,2);',
  9821. ' Insert(arrint[3],arrint,4);',
  9822. ' Insert(arrrec[5],arrrec,6);',
  9823. ' Insert(arrset[7],arrset,7);',
  9824. ' Insert(arrjsvalue[8],arrjsvalue,9);',
  9825. ' Insert(10,arrjsvalue,11);',
  9826. ' Insert([23],arrarrint,22);',
  9827. ' Delete(arrint,12,13);',
  9828. ' Delete(arrrec,14,15);',
  9829. ' Delete(arrset,17,18);',
  9830. ' Delete(arrjsvalue,19,10);']);
  9831. ConvertProgram;
  9832. CheckSource('TestArray_InsertDelete',
  9833. LinesToStr([ // statements
  9834. 'this.TFlag = {',
  9835. ' "0": "big",',
  9836. ' big: 0,',
  9837. ' "1": "small",',
  9838. ' small: 1',
  9839. '};',
  9840. 'rtl.recNewT(this, "TRec", function () {',
  9841. ' this.i = 0;',
  9842. ' this.$eq = function (b) {',
  9843. ' return this.i === b.i;',
  9844. ' };',
  9845. ' this.$assign = function (s) {',
  9846. ' this.i = s.i;',
  9847. ' return this;',
  9848. ' };',
  9849. '});',
  9850. 'this.ArrInt = [];',
  9851. 'this.ArrRec = [];',
  9852. 'this.ArrSet = [];',
  9853. 'this.ArrJSValue = [];',
  9854. 'this.ArrArrInt = [];',
  9855. '']),
  9856. LinesToStr([ // $mod.$main
  9857. '$mod.ArrInt.splice(2, 0, 1);',
  9858. '$mod.ArrInt.splice(4, 0, $mod.ArrInt[3]);',
  9859. '$mod.ArrRec.splice(6, 0, $mod.ArrRec[5]);',
  9860. '$mod.ArrSet.splice(7, 0, $mod.ArrSet[7]);',
  9861. '$mod.ArrJSValue.splice(9, 0, $mod.ArrJSValue[8]);',
  9862. '$mod.ArrJSValue.splice(11, 0, 10);',
  9863. '$mod.ArrArrInt.splice(22, 0, [23]);',
  9864. '$mod.ArrInt.splice(12, 13);',
  9865. '$mod.ArrRec.splice(14, 15);',
  9866. '$mod.ArrSet.splice(17, 18);',
  9867. '$mod.ArrJSValue.splice(19, 10);',
  9868. '']));
  9869. end;
  9870. procedure TTestModule.TestArray_DynArrayConstObjFPC;
  9871. begin
  9872. Parser.Options:=Parser.Options+[po_cassignments];
  9873. StartProgram(false);
  9874. Add([
  9875. '{$modeswitch arrayoperators}',
  9876. 'type',
  9877. ' integer = longint;',
  9878. ' TArrInt = array of integer;',
  9879. ' TArrStr = array of string;',
  9880. 'const',
  9881. ' Ints: TArrInt = (1,2,3);',
  9882. ' Aliases: TarrStr = (''foo'',''b'');',
  9883. ' OneInt: TArrInt = (7);',
  9884. ' OneStr: array of integer = (7);',
  9885. ' Chars: array of char = ''aoc'';',
  9886. ' Names: array of string = (''a'',''foo'');',
  9887. ' NameCount = low(Names)+high(Names)+length(Names);',
  9888. 'var i: integer;',
  9889. 'begin',
  9890. ' Ints:=[];',
  9891. ' Ints:=[1,1];',
  9892. ' Ints:=[1]+[2];',
  9893. ' Ints:=[2];',
  9894. ' Ints:=[]+ints;',
  9895. ' Ints:=Ints+[];',
  9896. ' Ints:=Ints+OneInt;',
  9897. ' Ints:=Ints+[1,1];',
  9898. ' Ints:=[i,i]+Ints;',
  9899. ' Ints:=[1]+[i]+[3];',
  9900. '']);
  9901. ConvertProgram;
  9902. CheckSource('TestArray_DynArrayConstObjFPC',
  9903. LinesToStr([ // statements
  9904. 'this.Ints = [1, 2, 3];',
  9905. 'this.Aliases = ["foo", "b"];',
  9906. 'this.OneInt = [7];',
  9907. 'this.OneStr = [7];',
  9908. 'this.Chars = ["a", "o", "c"];',
  9909. 'this.Names = ["a", "foo"];',
  9910. 'this.NameCount = 0 + (rtl.length(this.Names) - 1) + rtl.length(this.Names);',
  9911. 'this.i = 0;',
  9912. '']),
  9913. LinesToStr([ // $mod.$main
  9914. '$mod.Ints = [];',
  9915. '$mod.Ints = [1, 1];',
  9916. '$mod.Ints = rtl.arrayConcatN([1], [2]);',
  9917. '$mod.Ints = [2];',
  9918. '$mod.Ints = rtl.arrayConcatN([], $mod.Ints);',
  9919. '$mod.Ints = rtl.arrayConcatN($mod.Ints, []);',
  9920. '$mod.Ints = rtl.arrayConcatN($mod.Ints, $mod.OneInt);',
  9921. '$mod.Ints = rtl.arrayConcatN($mod.Ints, [1, 1]);',
  9922. '$mod.Ints = rtl.arrayConcatN([$mod.i, $mod.i], $mod.Ints);',
  9923. '$mod.Ints = rtl.arrayConcatN(rtl.arrayConcatN([1], [$mod.i]), [3]);',
  9924. '']));
  9925. end;
  9926. procedure TTestModule.TestArray_DynArrayConstDelphi;
  9927. begin
  9928. StartProgram(false);
  9929. // Note: const c = [1,1]; defines a set!
  9930. Add([
  9931. '{$mode delphi}',
  9932. 'type',
  9933. ' integer = longint;',
  9934. ' TArrInt = array of integer;',
  9935. ' TArrStr = array of string;',
  9936. 'const',
  9937. ' Ints: TArrInt = [1,1,2];',
  9938. ' Aliases: TarrStr = [''foo'',''b''];',
  9939. ' OneInt: TArrInt = [7];',
  9940. ' OneStr: array of integer = [7]+[8];',
  9941. ' Chars: array of char = ''aoc'';',
  9942. ' Names: array of string = [''a'',''a''];',
  9943. ' NameCount = low(Names)+high(Names)+length(Names);',
  9944. 'begin',
  9945. '']);
  9946. ConvertProgram;
  9947. CheckSource('TestArray_DynArrayConstDelphi',
  9948. LinesToStr([ // statements
  9949. 'this.Ints = [1, 1, 2];',
  9950. 'this.Aliases = ["foo", "b"];',
  9951. 'this.OneInt = [7];',
  9952. 'this.OneStr = rtl.arrayConcatN([7],[8]);',
  9953. 'this.Chars = ["a", "o", "c"];',
  9954. 'this.Names = ["a", "a"];',
  9955. 'this.NameCount = 0 + (rtl.length(this.Names) - 1) + rtl.length(this.Names);',
  9956. '']),
  9957. LinesToStr([ // $mod.$main
  9958. '']));
  9959. end;
  9960. procedure TTestModule.TestArray_ArrayLitAsParam;
  9961. begin
  9962. StartProgram(false);
  9963. Add([
  9964. '{$modeswitch arrayoperators}',
  9965. 'type',
  9966. ' integer = longint;',
  9967. ' TArrInt = array of integer;',
  9968. ' TArrSet = array of (red,green,blue);',
  9969. 'procedure DoOpenInt(const a: array of integer); forward;',
  9970. 'procedure DoInt(const a: TArrInt);',
  9971. 'begin',
  9972. ' DoInt(a+[1]);',
  9973. ' DoInt([1]+a);',
  9974. ' DoOpenInt(a);',
  9975. ' DoOpenInt(a+[1]);',
  9976. ' DoOpenInt([1]+a);',
  9977. 'end;',
  9978. 'procedure DoOpenInt(const a: array of integer);',
  9979. 'begin',
  9980. ' DoOpenInt(a+[1]);',
  9981. ' DoOpenInt([1]+a);',
  9982. ' DoInt(a);',
  9983. ' DoInt(a+[1]);',
  9984. ' DoInt([1]+a);',
  9985. 'end;',
  9986. 'procedure DoSet(const a: TArrSet);',
  9987. 'begin',
  9988. ' DoSet(a+[red]);',
  9989. ' DoSet([blue]+a);',
  9990. 'end;',
  9991. 'var',
  9992. ' i: TArrInt;',
  9993. ' s: TArrSet;',
  9994. 'begin',
  9995. ' DoInt([1]);',
  9996. ' DoInt([1]+[2]);',
  9997. ' DoInt(i+[1]);',
  9998. ' DoInt([1]+i);',
  9999. ' DoOpenInt([1]);',
  10000. ' DoOpenInt([1]+[2]);',
  10001. ' DoOpenInt(i+[1]);',
  10002. ' DoOpenInt([1]+i);',
  10003. ' DoSet([red]);',
  10004. ' DoSet([blue]+[green]);',
  10005. ' DoSet(s+[blue]);',
  10006. ' DoSet([red]+s);',
  10007. '']);
  10008. ConvertProgram;
  10009. CheckSource('TestArray_ArrayLitAsParam',
  10010. LinesToStr([ // statements
  10011. 'this.TArrSet$a = {',
  10012. ' "0": "red",',
  10013. ' red: 0,',
  10014. ' "1": "green",',
  10015. ' green: 1,',
  10016. ' "2": "blue",',
  10017. ' blue: 2',
  10018. '};',
  10019. 'this.DoInt = function (a) {',
  10020. ' $mod.DoInt(rtl.arrayConcatN(a, [1]));',
  10021. ' $mod.DoInt(rtl.arrayConcatN([1], a));',
  10022. ' $mod.DoOpenInt(a);',
  10023. ' $mod.DoOpenInt(rtl.arrayConcatN(a, [1]));',
  10024. ' $mod.DoOpenInt(rtl.arrayConcatN([1], a));',
  10025. '};',
  10026. 'this.DoOpenInt = function (a) {',
  10027. ' $mod.DoOpenInt(rtl.arrayConcatN(a, [1]));',
  10028. ' $mod.DoOpenInt(rtl.arrayConcatN([1], a));',
  10029. ' $mod.DoInt(a);',
  10030. ' $mod.DoInt(rtl.arrayConcatN(a, [1]));',
  10031. ' $mod.DoInt(rtl.arrayConcatN([1], a));',
  10032. '};',
  10033. 'this.DoSet = function (a) {',
  10034. ' $mod.DoSet(rtl.arrayConcatN(a, [$mod.TArrSet$a.red]));',
  10035. ' $mod.DoSet(rtl.arrayConcatN([$mod.TArrSet$a.blue], a));',
  10036. '};',
  10037. 'this.i = [];',
  10038. 'this.s = [];',
  10039. '']),
  10040. LinesToStr([ // $mod.$main
  10041. '$mod.DoInt([1]);',
  10042. '$mod.DoInt(rtl.arrayConcatN([1], [2]));',
  10043. '$mod.DoInt(rtl.arrayConcatN($mod.i, [1]));',
  10044. '$mod.DoInt(rtl.arrayConcatN([1], $mod.i));',
  10045. '$mod.DoOpenInt([1]);',
  10046. '$mod.DoOpenInt(rtl.arrayConcatN([1], [2]));',
  10047. '$mod.DoOpenInt(rtl.arrayConcatN($mod.i, [1]));',
  10048. '$mod.DoOpenInt(rtl.arrayConcatN([1], $mod.i));',
  10049. '$mod.DoSet([$mod.TArrSet$a.red]);',
  10050. '$mod.DoSet(rtl.arrayConcatN([$mod.TArrSet$a.blue], [$mod.TArrSet$a.green]));',
  10051. '$mod.DoSet(rtl.arrayConcatN($mod.s, [$mod.TArrSet$a.blue]));',
  10052. '$mod.DoSet(rtl.arrayConcatN([$mod.TArrSet$a.red], $mod.s));',
  10053. '']));
  10054. end;
  10055. procedure TTestModule.TestArray_ArrayLitMultiDimAsParam;
  10056. begin
  10057. StartProgram(false);
  10058. Add([
  10059. '{$modeswitch arrayoperators}',
  10060. 'type',
  10061. ' integer = longint;',
  10062. ' TArrInt = array of integer;',
  10063. ' TArrArrInt = array of TArrInt;',
  10064. 'procedure DoInt(const a: TArrArrInt);',
  10065. 'begin',
  10066. ' DoInt(a+[[1]]);',
  10067. ' DoInt([[1]]+a);',
  10068. ' DoInt(a);',
  10069. 'end;',
  10070. 'var',
  10071. ' i: TArrInt;',
  10072. ' a: TArrArrInt;',
  10073. 'begin',
  10074. ' a:=[[1]];',
  10075. ' a:=[i];',
  10076. ' a:=a+[i];',
  10077. ' a:=[i]+a;',
  10078. ' a:=[[1]+i];',
  10079. ' a:=[[1]+[2]];',
  10080. ' a:=[i+[2]];',
  10081. ' DoInt([[1]]);',
  10082. ' DoInt([[1]+[2],[3,4],[5]]);',
  10083. ' DoInt([i+[1]]+a);',
  10084. ' DoInt([i]+a);',
  10085. '']);
  10086. ConvertProgram;
  10087. CheckSource('TestArray_ArrayLitMultiDimAsParam',
  10088. LinesToStr([ // statements
  10089. 'this.DoInt = function (a) {',
  10090. ' $mod.DoInt(rtl.arrayConcatN(a, [[1]]));',
  10091. ' $mod.DoInt(rtl.arrayConcatN([[1]], a));',
  10092. ' $mod.DoInt(a);',
  10093. '};',
  10094. 'this.i = [];',
  10095. 'this.a = [];',
  10096. '']),
  10097. LinesToStr([ // $mod.$main
  10098. '$mod.a = [[1]];',
  10099. '$mod.a = [$mod.i];',
  10100. '$mod.a = rtl.arrayConcatN($mod.a, [$mod.i]);',
  10101. '$mod.a = rtl.arrayConcatN([$mod.i], $mod.a);',
  10102. '$mod.a = [rtl.arrayConcatN([1], $mod.i)];',
  10103. '$mod.a = [rtl.arrayConcatN([1], [2])];',
  10104. '$mod.a = [rtl.arrayConcatN($mod.i, [2])];',
  10105. '$mod.DoInt([[1]]);',
  10106. '$mod.DoInt([rtl.arrayConcatN([1], [2]), [3, 4], [5]]);',
  10107. '$mod.DoInt(rtl.arrayConcatN([rtl.arrayConcatN($mod.i, [1])], $mod.a));',
  10108. '$mod.DoInt(rtl.arrayConcatN([$mod.i], $mod.a));',
  10109. '']));
  10110. end;
  10111. procedure TTestModule.TestArray_ArrayLitStaticAsParam;
  10112. begin
  10113. StartProgram(false);
  10114. Add([
  10115. '{$modeswitch arrayoperators}',
  10116. 'type',
  10117. ' integer = longint;',
  10118. ' TArrInt = array[1..2] of integer;',
  10119. ' TArrArrInt = array of TArrInt;',
  10120. 'procedure DoInt(const a: TArrArrInt);',
  10121. 'begin',
  10122. ' DoInt(a+[[1,2]]);',
  10123. ' DoInt([[1,2]]+a);',
  10124. ' DoInt(a);',
  10125. 'end;',
  10126. 'var',
  10127. ' i: TArrInt;',
  10128. ' a: TArrArrInt;',
  10129. 'begin',
  10130. ' a:=[[1,1]];',
  10131. ' a:=[i];',
  10132. ' a:=a+[i];',
  10133. ' a:=[i]+a;',
  10134. ' DoInt([[1,1]]);',
  10135. ' DoInt([[1,2],[3,4]]);',
  10136. '']);
  10137. ConvertProgram;
  10138. CheckSource('TestArray_ArrayLitStaticAsParam',
  10139. LinesToStr([ // statements
  10140. 'this.DoInt = function (a) {',
  10141. ' $mod.DoInt(rtl.arrayConcatN(a, [[1, 2]]));',
  10142. ' $mod.DoInt(rtl.arrayConcatN([[1, 2]], a));',
  10143. ' $mod.DoInt(a);',
  10144. '};',
  10145. 'this.i = rtl.arraySetLength(null, 0, 2);',
  10146. 'this.a = [];',
  10147. '']),
  10148. LinesToStr([ // $mod.$main
  10149. '$mod.a = [[1, 1]];',
  10150. '$mod.a = [$mod.i.slice(0)];',
  10151. '$mod.a = rtl.arrayConcatN($mod.a, [$mod.i.slice(0)]);',
  10152. '$mod.a = rtl.arrayConcatN([$mod.i.slice(0)], $mod.a);',
  10153. '$mod.DoInt([[1, 1]]);',
  10154. '$mod.DoInt([[1, 2], [3, 4]]);',
  10155. '']));
  10156. end;
  10157. procedure TTestModule.TestArray_ForInArrOfString;
  10158. begin
  10159. StartProgram(false);
  10160. Add([
  10161. 'type',
  10162. 'type',
  10163. ' TMonthNameArray = array [1..12] of string;',
  10164. ' TMonthNames = TMonthNameArray;',
  10165. ' TObject = class',
  10166. ' private',
  10167. ' function GetLongMonthNames: TMonthNames; virtual; abstract;',
  10168. ' public',
  10169. ' Property LongMonthNames : TMonthNames Read GetLongMonthNames;',
  10170. ' end;',
  10171. 'var',
  10172. ' f: TObject;',
  10173. ' Month: string;',
  10174. ' Names: array of string = (''a'',''foo'',''bar'');',
  10175. ' i: longint;',
  10176. 'begin',
  10177. ' for Month in f.LongMonthNames do ;',
  10178. ' for Month in Names do ;',
  10179. ' for i:=low(Names) to high(Names) do ;',
  10180. '']);
  10181. ConvertProgram;
  10182. CheckSource('TestArray_ForInArrOfString',
  10183. LinesToStr([ // statements
  10184. 'rtl.createClass(this, "TObject", null, function () {',
  10185. ' this.$init = function () {',
  10186. ' };',
  10187. ' this.$final = function () {',
  10188. ' };',
  10189. '});',
  10190. 'this.f = null;',
  10191. 'this.Month = "";',
  10192. 'this.Names = ["a", "foo", "bar"];',
  10193. 'this.i = 0;',
  10194. '']),
  10195. LinesToStr([ // $mod.$main
  10196. 'for (var $in = $mod.f.GetLongMonthNames(), $l = 0, $end = rtl.length($in) - 1; $l <= $end; $l++) $mod.Month = $in[$l];',
  10197. 'for (var $in1 = $mod.Names, $l1 = 0, $end1 = rtl.length($in1) - 1; $l1 <= $end1; $l1++) $mod.Month = $in1[$l1];',
  10198. 'for (var $l2 = 0, $end2 = rtl.length($mod.Names) - 1; $l2 <= $end2; $l2++) $mod.i = $l2;',
  10199. '']));
  10200. end;
  10201. procedure TTestModule.TestExternalClass_TypeCastArrayToExternalClass;
  10202. begin
  10203. StartProgram(false);
  10204. Add([
  10205. '{$modeswitch externalclass}',
  10206. 'type',
  10207. ' TJSObject = class external name ''Object''',
  10208. ' end;',
  10209. ' TJSArray = class external name ''Array''',
  10210. ' class function isArray(Value: JSValue) : boolean;',
  10211. ' function concat() : TJSArray; varargs;',
  10212. ' end;',
  10213. 'var',
  10214. ' aObj: TJSArray;',
  10215. ' a: array of longint;',
  10216. ' o: TJSObject;',
  10217. 'begin',
  10218. ' if TJSArray.isArray(65) then ;',
  10219. ' aObj:=TJSArray(a).concat(a);',
  10220. ' o:=TJSObject(a);']);
  10221. ConvertProgram;
  10222. CheckSource('TestExternalClass_TypeCastArrayToExternalClass',
  10223. LinesToStr([ // statements
  10224. 'this.aObj = null;',
  10225. 'this.a = [];',
  10226. 'this.o = null;',
  10227. '']),
  10228. LinesToStr([ // $mod.$main
  10229. 'if (Array.isArray(65)) ;',
  10230. '$mod.aObj = $mod.a.concat($mod.a);',
  10231. '$mod.o = $mod.a;',
  10232. '']));
  10233. end;
  10234. procedure TTestModule.TestExternalClass_TypeCastArrayFromExternalClass;
  10235. begin
  10236. StartProgram(false);
  10237. Add([
  10238. '{$modeswitch externalclass}',
  10239. 'type',
  10240. ' TArrStr = array of string;',
  10241. ' TJSArray = class external name ''Array''',
  10242. ' end;',
  10243. ' TJSObject = class external name ''Object''',
  10244. ' end;',
  10245. 'var',
  10246. ' aObj: TJSArray;',
  10247. ' a: TArrStr;',
  10248. ' jo: TJSObject;',
  10249. 'begin',
  10250. ' a:=TArrStr(aObj);',
  10251. ' TArrStr(aObj)[1]:=TArrStr(aObj)[2];',
  10252. ' a:=TarrStr(jo);',
  10253. '']);
  10254. ConvertProgram;
  10255. CheckSource('TestExternalClass_TypeCastArrayFromExternalClass',
  10256. LinesToStr([ // statements
  10257. 'this.aObj = null;',
  10258. 'this.a = [];',
  10259. 'this.jo = null;',
  10260. '']),
  10261. LinesToStr([ // $mod.$main
  10262. '$mod.a = $mod.aObj;',
  10263. '$mod.aObj[1] = $mod.aObj[2];',
  10264. '$mod.a = $mod.jo;',
  10265. '']));
  10266. end;
  10267. procedure TTestModule.TestArrayOfConst_TVarRec;
  10268. begin
  10269. StartProgram(true,[supTVarRec]);
  10270. Add([
  10271. 'procedure Say(args: array of const);',
  10272. 'var',
  10273. ' i: longint;',
  10274. ' v: TVarRec;',
  10275. 'begin',
  10276. ' for i:=low(args) to high(args) do begin',
  10277. ' v:=args[i];',
  10278. ' case v.vtype of',
  10279. ' vtInteger: if length(args)=args[i].vInteger then ;',
  10280. ' end;',
  10281. ' end;',
  10282. ' for v in args do ;',
  10283. ' args:=nil;',
  10284. ' SetLength(args,2);',
  10285. 'end;',
  10286. 'begin']);
  10287. ConvertProgram;
  10288. CheckSource('TestArrayOfConst_TVarRec',
  10289. LinesToStr([ // statements
  10290. 'this.Say = function (args) {',
  10291. ' var i = 0;',
  10292. ' var v = pas.system.TVarRec.$new();',
  10293. ' for (var $l = 0, $end = rtl.length(args) - 1; $l <= $end; $l++) {',
  10294. ' i = $l;',
  10295. ' v.$assign(args[i]);',
  10296. ' var $tmp = v.VType;',
  10297. ' if ($tmp === 0) if (rtl.length(args) === args[i].VJSValue) ;',
  10298. ' };',
  10299. ' for (var $in = args, $l1 = 0, $end1 = rtl.length($in) - 1; $l1 <= $end1; $l1++) v = $in[$l1];',
  10300. ' args = [];',
  10301. ' args = rtl.arraySetLength(args, pas.system.TVarRec, 2);',
  10302. '};',
  10303. '']),
  10304. LinesToStr([ // $mod.$main
  10305. ]));
  10306. end;
  10307. procedure TTestModule.TestArrayOfConst_PassBaseTypes;
  10308. begin
  10309. StartProgram(true,[supTVarRec]);
  10310. Add([
  10311. 'procedure Say(args: array of const);',
  10312. 'begin',
  10313. ' Say(args);',
  10314. 'end;',
  10315. 'var',
  10316. ' p: Pointer;',
  10317. ' j: jsvalue;',
  10318. ' c: currency;',
  10319. 'begin',
  10320. ' Say([]);',
  10321. ' Say([1]);',
  10322. ' Say([''c'',''foo'',nil,true,1.3,p,j,c]);',
  10323. '']);
  10324. ConvertProgram;
  10325. CheckSource('TestArrayOfConst_PassBaseTypes',
  10326. LinesToStr([ // statements
  10327. 'this.Say = function (args) {',
  10328. ' $mod.Say(args);',
  10329. '};',
  10330. 'this.p = null;',
  10331. 'this.j = undefined;',
  10332. 'this.c = 0;',
  10333. '']),
  10334. LinesToStr([ // $mod.$main
  10335. '$mod.Say([]);',
  10336. '$mod.Say(pas.system.VarRecs(0, 1));',
  10337. '$mod.Say(pas.system.VarRecs(',
  10338. ' 9,',
  10339. ' "c",',
  10340. ' 18,',
  10341. ' "foo",',
  10342. ' 5,',
  10343. ' null,',
  10344. ' 1,',
  10345. ' true,',
  10346. ' 3,',
  10347. ' 1.3,',
  10348. ' 5,',
  10349. ' $mod.p,',
  10350. ' 20,',
  10351. ' $mod.j,',
  10352. ' 12,',
  10353. ' $mod.c',
  10354. ' ));',
  10355. '']));
  10356. end;
  10357. procedure TTestModule.TestArrayOfConst_PassObj;
  10358. begin
  10359. StartProgram(true,[supTVarRec]);
  10360. Add([
  10361. '{$interfaces corba}',
  10362. 'type',
  10363. ' TObject = class',
  10364. ' end;',
  10365. ' TClass = class of TObject;',
  10366. ' IUnknown = interface',
  10367. ' end;',
  10368. 'procedure Say(args: array of const);',
  10369. 'begin',
  10370. 'end;',
  10371. 'var',
  10372. ' o: TObject;',
  10373. ' c: TClass;',
  10374. ' i: IUnknown;',
  10375. 'begin',
  10376. ' Say([o,c,TObject]);',
  10377. ' Say([nil,i]);',
  10378. '']);
  10379. ConvertProgram;
  10380. CheckSource('TestArrayOfConst_PassObj',
  10381. LinesToStr([ // statements
  10382. 'rtl.createClass(this, "TObject", null, function () {',
  10383. ' this.$init = function () {',
  10384. ' };',
  10385. ' this.$final = function () {',
  10386. ' };',
  10387. '});',
  10388. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  10389. 'this.Say = function (args) {',
  10390. '};',
  10391. 'this.o = null;',
  10392. 'this.c = null;',
  10393. 'this.i = null;',
  10394. '']),
  10395. LinesToStr([ // $mod.$main
  10396. '$mod.Say(pas.system.VarRecs(',
  10397. ' 7,',
  10398. ' $mod.o,',
  10399. ' 8,',
  10400. ' $mod.c,',
  10401. ' 8,',
  10402. ' $mod.TObject',
  10403. '));',
  10404. '$mod.Say(pas.system.VarRecs(5, null, 14, $mod.i));',
  10405. '']));
  10406. end;
  10407. procedure TTestModule.TestRecord_Empty;
  10408. begin
  10409. StartProgram(false);
  10410. Add([
  10411. 'type',
  10412. ' TRecA = record',
  10413. ' end;',
  10414. 'var a,b: TRecA;',
  10415. 'begin',
  10416. ' if a=b then ;']);
  10417. ConvertProgram;
  10418. CheckSource('TestRecord_Empty',
  10419. LinesToStr([ // statements
  10420. 'rtl.recNewT(this, "TRecA", function () {',
  10421. ' this.$eq = function (b) {',
  10422. ' return true;',
  10423. ' };',
  10424. ' this.$assign = function (s) {',
  10425. ' return this;',
  10426. ' };',
  10427. '});',
  10428. 'this.a = this.TRecA.$new();',
  10429. 'this.b = this.TRecA.$new();',
  10430. '']),
  10431. LinesToStr([ // $mod.$main
  10432. 'if ($mod.a.$eq($mod.b)) ;'
  10433. ]));
  10434. end;
  10435. procedure TTestModule.TestRecord_Var;
  10436. begin
  10437. StartProgram(false);
  10438. Add('type');
  10439. Add(' TRecA = record');
  10440. Add(' Bold: longint;');
  10441. Add(' end;');
  10442. Add('var Rec: TRecA;');
  10443. Add('begin');
  10444. Add(' rec.bold:=123');
  10445. ConvertProgram;
  10446. CheckSource('TestRecord_Var',
  10447. LinesToStr([ // statements
  10448. 'rtl.recNewT(this, "TRecA", function () {',
  10449. ' this.Bold = 0;',
  10450. ' this.$eq = function (b) {',
  10451. ' return this.Bold === b.Bold;',
  10452. ' };',
  10453. ' this.$assign = function (s) {',
  10454. ' this.Bold = s.Bold;',
  10455. ' return this;',
  10456. ' };',
  10457. '});',
  10458. 'this.Rec = this.TRecA.$new();',
  10459. '']),
  10460. LinesToStr([ // $mod.$main
  10461. '$mod.Rec.Bold = 123;'
  10462. ]));
  10463. end;
  10464. procedure TTestModule.TestRecord_VarExternal;
  10465. begin
  10466. StartProgram(false);
  10467. Add([
  10468. '{$modeswitch externalclass}',
  10469. 'type',
  10470. ' TRecA = record',
  10471. ' i: byte;',
  10472. ' length_: longint external name ''length'';',
  10473. ' end;',
  10474. 'var Rec: TRecA;',
  10475. 'begin',
  10476. ' rec.length_ := rec.length_',
  10477. '']);
  10478. ConvertProgram;
  10479. CheckSource('TestRecord_VarExternal',
  10480. LinesToStr([ // statements
  10481. 'rtl.recNewT(this, "TRecA", function () {',
  10482. ' this.i = 0;',
  10483. ' this.$eq = function (b) {',
  10484. ' return (this.i === b.i) && (this.length === b.length);',
  10485. ' };',
  10486. ' this.$assign = function (s) {',
  10487. ' this.i = s.i;',
  10488. ' this.length = s.length;',
  10489. ' return this;',
  10490. ' };',
  10491. '});',
  10492. 'this.Rec = this.TRecA.$new();',
  10493. '']),
  10494. LinesToStr([ // $mod.$main
  10495. '$mod.Rec.length = $mod.Rec.length;'
  10496. ]));
  10497. end;
  10498. procedure TTestModule.TestRecord_WithDo;
  10499. begin
  10500. StartProgram(false);
  10501. Add('type');
  10502. Add(' TRec = record');
  10503. Add(' vI: longint;');
  10504. Add(' end;');
  10505. Add('var');
  10506. Add(' Int: longint;');
  10507. Add(' r: TRec;');
  10508. Add('begin');
  10509. Add(' with r do');
  10510. Add(' int:=vi;');
  10511. Add(' with r do begin');
  10512. Add(' int:=vi;');
  10513. Add(' vi:=int;');
  10514. Add(' end;');
  10515. ConvertProgram;
  10516. CheckSource('TestWithRecordDo',
  10517. LinesToStr([ // statements
  10518. 'rtl.recNewT(this, "TRec", function () {',
  10519. ' this.vI = 0;',
  10520. ' this.$eq = function (b) {',
  10521. ' return this.vI === b.vI;',
  10522. ' };',
  10523. ' this.$assign = function (s) {',
  10524. ' this.vI = s.vI;',
  10525. ' return this;',
  10526. ' };',
  10527. '});',
  10528. 'this.Int = 0;',
  10529. 'this.r = this.TRec.$new();',
  10530. '']),
  10531. LinesToStr([ // $mod.$main
  10532. 'var $with = $mod.r;',
  10533. '$mod.Int = $with.vI;',
  10534. 'var $with1 = $mod.r;',
  10535. '$mod.Int = $with1.vI;',
  10536. '$with1.vI = $mod.Int;'
  10537. ]));
  10538. end;
  10539. procedure TTestModule.TestRecord_Assign;
  10540. begin
  10541. StartProgram(false);
  10542. Add('type');
  10543. Add(' TEnum = (red,green);');
  10544. Add(' TEnums = set of TEnum;');
  10545. Add(' TSmallRec = record');
  10546. Add(' N: longint;');
  10547. Add(' end;');
  10548. Add(' TBigRec = record');
  10549. Add(' Int: longint;');
  10550. Add(' D: double;');
  10551. Add(' Arr: array of longint;');
  10552. Add(' Arr2: array[1..2] of longint;');
  10553. Add(' Small: TSmallRec;');
  10554. Add(' Enums: TEnums;');
  10555. Add(' end;');
  10556. Add('var');
  10557. Add(' r, s: TBigRec;');
  10558. Add('begin');
  10559. Add(' r:=s;');
  10560. Add(' r:=default(TBigRec);');
  10561. Add(' r:=default(s);');
  10562. ConvertProgram;
  10563. CheckSource('TestRecord_Assign',
  10564. LinesToStr([ // statements
  10565. 'this.TEnum = {',
  10566. ' "0": "red",',
  10567. ' red: 0,',
  10568. ' "1": "green",',
  10569. ' green: 1',
  10570. '};',
  10571. 'rtl.recNewT(this, "TSmallRec", function () {',
  10572. ' this.N = 0;',
  10573. ' this.$eq = function (b) {',
  10574. ' return this.N === b.N;',
  10575. ' };',
  10576. ' this.$assign = function (s) {',
  10577. ' this.N = s.N;',
  10578. ' return this;',
  10579. ' };',
  10580. '});',
  10581. 'rtl.recNewT(this, "TBigRec", function () {',
  10582. ' this.Int = 0;',
  10583. ' this.D = 0.0;',
  10584. ' this.$new = function () {',
  10585. ' var r = Object.create(this);',
  10586. ' r.Arr = [];',
  10587. ' r.Arr2 = rtl.arraySetLength(null, 0, 2);',
  10588. ' r.Small = $mod.TSmallRec.$new();',
  10589. ' r.Enums = {};',
  10590. ' return r;',
  10591. ' };',
  10592. ' this.$eq = function (b) {',
  10593. ' return (this.Int === b.Int) && (this.D === b.D) && (this.Arr === b.Arr) && rtl.arrayEq(this.Arr2, b.Arr2) && this.Small.$eq(b.Small) && rtl.eqSet(this.Enums, b.Enums);',
  10594. ' };',
  10595. ' this.$assign = function (s) {',
  10596. ' this.Int = s.Int;',
  10597. ' this.D = s.D;',
  10598. ' this.Arr = rtl.arrayRef(s.Arr);',
  10599. ' this.Arr2 = s.Arr2.slice(0);',
  10600. ' this.Small.$assign(s.Small);',
  10601. ' this.Enums = rtl.refSet(s.Enums);',
  10602. ' return this;',
  10603. ' };',
  10604. '});',
  10605. 'this.r = this.TBigRec.$new();',
  10606. 'this.s = this.TBigRec.$new();',
  10607. '']),
  10608. LinesToStr([ // $mod.$main
  10609. '$mod.r.$assign($mod.s);',
  10610. '$mod.r.$assign($mod.TBigRec.$new());',
  10611. '$mod.r.$assign($mod.TBigRec.$new());',
  10612. '']));
  10613. end;
  10614. procedure TTestModule.TestRecord_AsParams;
  10615. begin
  10616. StartProgram(false);
  10617. Add([
  10618. 'type',
  10619. ' integer = longint;',
  10620. ' TRecord = record',
  10621. ' i: integer;',
  10622. ' end;',
  10623. 'procedure DoIt(vD: TRecord; const vC: TRecord; var vV: TRecord; var U);',
  10624. 'var vL: TRecord;',
  10625. 'begin',
  10626. ' vd:=vd;',
  10627. ' vd.i:=vd.i;',
  10628. ' vl:=vc;',
  10629. ' vv:=vv;',
  10630. ' vv.i:=vv.i;',
  10631. ' U:=vl;',
  10632. ' U:=vd;',
  10633. ' U:=vc;',
  10634. ' U:=vv;',
  10635. ' vl:=TRecord(U);',
  10636. ' vd:=TRecord(U);',
  10637. ' vv:=TRecord(U);',
  10638. ' doit(vd,vd,vd,vd);',
  10639. ' doit(vc,vc,vl,vl);',
  10640. ' doit(vv,vv,vv,vv);',
  10641. ' doit(vl,vl,vl,vl);',
  10642. ' TRecord(U).i:=3;',
  10643. 'end;',
  10644. 'var i: TRecord;',
  10645. 'begin',
  10646. ' doit(i,i,i,i);',
  10647. '']);
  10648. ConvertProgram;
  10649. CheckSource('TestRecord_AsParams',
  10650. LinesToStr([ // statements
  10651. 'rtl.recNewT(this, "TRecord", function () {',
  10652. ' this.i = 0;',
  10653. ' this.$eq = function (b) {',
  10654. ' return this.i === b.i;',
  10655. ' };',
  10656. ' this.$assign = function (s) {',
  10657. ' this.i = s.i;',
  10658. ' return this;',
  10659. ' };',
  10660. '});',
  10661. 'this.DoIt = function (vD, vC, vV, U) {',
  10662. ' var vL = $mod.TRecord.$new();',
  10663. ' vD.$assign(vD);',
  10664. ' vD.i = vD.i;',
  10665. ' vL.$assign(vC);',
  10666. ' vV.$assign(vV);',
  10667. ' vV.i = vV.i;',
  10668. ' U.$assign(vL);',
  10669. ' U.$assign(vD);',
  10670. ' U.$assign(vC);',
  10671. ' U.$assign(vV);',
  10672. ' vL.$assign(U);',
  10673. ' vD.$assign(U);',
  10674. ' vV.$assign(U);',
  10675. ' $mod.DoIt($mod.TRecord.$clone(vD), vD, vD, vD);',
  10676. ' $mod.DoIt($mod.TRecord.$clone(vC), vC, vL, vL);',
  10677. ' $mod.DoIt($mod.TRecord.$clone(vV), vV, vV, vV);',
  10678. ' $mod.DoIt($mod.TRecord.$clone(vL), vL, vL, vL);',
  10679. ' U.i = 3;',
  10680. '};',
  10681. 'this.i = this.TRecord.$new();'
  10682. ]),
  10683. LinesToStr([
  10684. '$mod.DoIt($mod.TRecord.$clone($mod.i), $mod.i, $mod.i, $mod.i);',
  10685. '']));
  10686. end;
  10687. procedure TTestModule.TestRecord_ConstRef;
  10688. begin
  10689. StartProgram(false);
  10690. Add([
  10691. 'type TRec = record i: word; end;',
  10692. 'procedure Run(constref a: TRec);',
  10693. 'begin',
  10694. 'end;',
  10695. 'procedure Fly(a: TRec; var b: TRec; out c: TRec; const d: TRec; constref e: TRec);',
  10696. 'var l: TRec;',
  10697. 'begin',
  10698. ' Run(l);',
  10699. ' Run(a);',
  10700. ' Run(b);',
  10701. ' Run(c);',
  10702. ' Run(d);',
  10703. ' Run(e);',
  10704. 'end;',
  10705. 'begin',
  10706. '']);
  10707. ConvertProgram;
  10708. CheckResolverUnexpectedHints();
  10709. CheckSource('TestRecord_ConstRef',
  10710. LinesToStr([ // statements
  10711. 'rtl.recNewT(this, "TRec", function () {',
  10712. ' this.i = 0;',
  10713. ' this.$eq = function (b) {',
  10714. ' return this.i === b.i;',
  10715. ' };',
  10716. ' this.$assign = function (s) {',
  10717. ' this.i = s.i;',
  10718. ' return this;',
  10719. ' };',
  10720. '});',
  10721. 'this.Run = function (a) {',
  10722. '};',
  10723. 'this.Fly = function (a, b, c, d, e) {',
  10724. ' var l = $mod.TRec.$new();',
  10725. ' $mod.Run(l);',
  10726. ' $mod.Run(a);',
  10727. ' $mod.Run(b);',
  10728. ' $mod.Run(c);',
  10729. ' $mod.Run(d);',
  10730. ' $mod.Run(e);',
  10731. '};',
  10732. '']),
  10733. LinesToStr([
  10734. '']));
  10735. end;
  10736. procedure TTestModule.TestRecordElement_AsParams;
  10737. begin
  10738. StartProgram(false);
  10739. Add('type');
  10740. Add(' integer = longint;');
  10741. Add(' TRecord = record');
  10742. Add(' i: integer;');
  10743. Add(' end;');
  10744. Add('procedure DoIt(vG: integer; const vH: integer; var vI: integer);');
  10745. Add('var vJ: TRecord;');
  10746. Add('begin');
  10747. Add(' doit(vj.i,vj.i,vj.i);');
  10748. Add('end;');
  10749. Add('var r: TRecord;');
  10750. Add('begin');
  10751. Add(' doit(r.i,r.i,r.i);');
  10752. ConvertProgram;
  10753. CheckSource('TestRecordElement_AsParams',
  10754. LinesToStr([ // statements
  10755. 'rtl.recNewT(this, "TRecord", function () {',
  10756. ' this.i = 0;',
  10757. ' this.$eq = function (b) {',
  10758. ' return this.i === b.i;',
  10759. ' };',
  10760. ' this.$assign = function (s) {',
  10761. ' this.i = s.i;',
  10762. ' return this;',
  10763. ' };',
  10764. '});',
  10765. 'this.DoIt = function (vG,vH,vI) {',
  10766. ' var vJ = $mod.TRecord.$new();',
  10767. ' $mod.DoIt(vJ.i, vJ.i, {',
  10768. ' p: vJ,',
  10769. ' get: function () {',
  10770. ' return this.p.i;',
  10771. ' },',
  10772. ' set: function (v) {',
  10773. ' this.p.i = v;',
  10774. ' }',
  10775. ' });',
  10776. '};',
  10777. 'this.r = this.TRecord.$new();'
  10778. ]),
  10779. LinesToStr([
  10780. '$mod.DoIt($mod.r.i,$mod.r.i,{',
  10781. ' p: $mod.r,',
  10782. ' get: function () {',
  10783. ' return this.p.i;',
  10784. ' },',
  10785. ' set: function (v) {',
  10786. ' this.p.i = v;',
  10787. ' }',
  10788. '});'
  10789. ]));
  10790. end;
  10791. procedure TTestModule.TestRecordElementFromFuncResult_AsParams;
  10792. begin
  10793. StartProgram(false);
  10794. Add('type');
  10795. Add(' integer = longint;');
  10796. Add(' TRecord = record');
  10797. Add(' i: integer;');
  10798. Add(' end;');
  10799. Add('function GetRec(vB: integer = 0): TRecord;');
  10800. Add('begin');
  10801. Add('end;');
  10802. Add('procedure DoIt(vG: integer; const vH: integer);');
  10803. Add('begin');
  10804. Add('end;');
  10805. Add('begin');
  10806. Add(' doit(getrec.i,getrec.i);');
  10807. Add(' doit(getrec().i,getrec().i);');
  10808. Add(' doit(getrec(1).i,getrec(2).i);');
  10809. ConvertProgram;
  10810. CheckSource('TestRecordElementFromFuncResult_AsParams',
  10811. LinesToStr([ // statements
  10812. 'rtl.recNewT(this, "TRecord", function () {',
  10813. ' this.i = 0;',
  10814. ' this.$eq = function (b) {',
  10815. ' return this.i === b.i;',
  10816. ' };',
  10817. ' this.$assign = function (s) {',
  10818. ' this.i = s.i;',
  10819. ' return this;',
  10820. ' };',
  10821. '});',
  10822. 'this.GetRec = function (vB) {',
  10823. ' var Result = $mod.TRecord.$new();',
  10824. ' return Result;',
  10825. '};',
  10826. 'this.DoIt = function (vG, vH) {',
  10827. '};',
  10828. '']),
  10829. LinesToStr([
  10830. '$mod.DoIt($mod.GetRec(0).i,$mod.GetRec(0).i);',
  10831. '$mod.DoIt($mod.GetRec(0).i,$mod.GetRec(0).i);',
  10832. '$mod.DoIt($mod.GetRec(1).i,$mod.GetRec(2).i);',
  10833. '']));
  10834. end;
  10835. procedure TTestModule.TestRecordElementFromWith_AsParams;
  10836. begin
  10837. StartProgram(false);
  10838. Add('type');
  10839. Add(' integer = longint;');
  10840. Add(' TRecord = record');
  10841. Add(' i: integer;');
  10842. Add(' end;');
  10843. Add('procedure DoIt(vG: integer; const vH: integer; var vI: integer);');
  10844. Add('begin');
  10845. Add('end;');
  10846. Add('var r: trecord;');
  10847. Add('begin');
  10848. Add(' with r do ');
  10849. Add(' doit(i,i,i);');
  10850. ConvertProgram;
  10851. CheckSource('TestRecordElementFromWith_AsParams',
  10852. LinesToStr([ // statements
  10853. 'rtl.recNewT(this, "TRecord", function () {',
  10854. ' this.i = 0;',
  10855. ' this.$eq = function (b) {',
  10856. ' return this.i === b.i;',
  10857. ' };',
  10858. ' this.$assign = function (s) {',
  10859. ' this.i = s.i;',
  10860. ' return this;',
  10861. ' };',
  10862. '});',
  10863. 'this.DoIt = function (vG,vH,vI) {',
  10864. '};',
  10865. 'this.r = this.TRecord.$new();'
  10866. ]),
  10867. LinesToStr([
  10868. 'var $with = $mod.r;',
  10869. '$mod.DoIt($with.i,$with.i,{',
  10870. ' p: $with,',
  10871. ' get: function () {',
  10872. ' return this.p.i;',
  10873. ' },',
  10874. ' set: function (v) {',
  10875. ' this.p.i = v;',
  10876. ' }',
  10877. '});',
  10878. '']));
  10879. end;
  10880. procedure TTestModule.TestRecord_Equal;
  10881. begin
  10882. StartProgram(false);
  10883. Add('type');
  10884. Add(' integer = longint;');
  10885. Add(' TFlag = (red,blue);');
  10886. Add(' TFlags = set of TFlag;');
  10887. Add(' TProc = procedure;');
  10888. Add(' TRecord = record');
  10889. Add(' i: integer;');
  10890. Add(' Event: TProc;');
  10891. Add(' f: TFlags;');
  10892. Add(' end;');
  10893. Add(' TNested = record');
  10894. Add(' r: TRecord;');
  10895. Add(' end;');
  10896. Add('var');
  10897. Add(' b: boolean;');
  10898. Add(' r,s: trecord;');
  10899. Add('begin');
  10900. Add(' b:=r=s;');
  10901. Add(' b:=r<>s;');
  10902. ConvertProgram;
  10903. CheckSource('TestRecord_Equal',
  10904. LinesToStr([ // statements
  10905. 'this.TFlag = {',
  10906. ' "0": "red",',
  10907. ' red: 0,',
  10908. ' "1": "blue",',
  10909. ' blue: 1',
  10910. '};',
  10911. 'rtl.recNewT(this, "TRecord", function () {',
  10912. ' this.i = 0;',
  10913. ' this.Event = null;',
  10914. ' this.$new = function () {',
  10915. ' var r = Object.create(this);',
  10916. ' r.f = {};',
  10917. ' return r;',
  10918. ' };',
  10919. ' this.$eq = function (b) {',
  10920. ' return (this.i === b.i) && rtl.eqCallback(this.Event, b.Event) && rtl.eqSet(this.f, b.f);',
  10921. ' };',
  10922. ' this.$assign = function (s) {',
  10923. ' this.i = s.i;',
  10924. ' this.Event = s.Event;',
  10925. ' this.f = rtl.refSet(s.f);',
  10926. ' return this;',
  10927. ' };',
  10928. '});',
  10929. 'rtl.recNewT(this, "TNested", function () {',
  10930. ' this.$new = function () {',
  10931. ' var r = Object.create(this);',
  10932. ' r.r = $mod.TRecord.$new();',
  10933. ' return r;',
  10934. ' };',
  10935. ' this.$eq = function (b) {',
  10936. ' return this.r.$eq(b.r);',
  10937. ' };',
  10938. ' this.$assign = function (s) {',
  10939. ' this.r.$assign(s.r);',
  10940. ' return this;',
  10941. ' };',
  10942. '});',
  10943. 'this.b = false;',
  10944. 'this.r = this.TRecord.$new();',
  10945. 'this.s = this.TRecord.$new();',
  10946. '']),
  10947. LinesToStr([
  10948. '$mod.b = $mod.r.$eq($mod.s);',
  10949. '$mod.b = !$mod.r.$eq($mod.s);',
  10950. '']));
  10951. end;
  10952. procedure TTestModule.TestRecord_JSValue;
  10953. begin
  10954. StartProgram(false);
  10955. Add([
  10956. 'type',
  10957. ' TRecord = record',
  10958. ' i: longint;',
  10959. ' end;',
  10960. 'procedure Fly(d: jsvalue; const c: jsvalue);',
  10961. 'begin',
  10962. 'end;',
  10963. 'procedure Run(d: TRecord; const c: TRecord; var v: TRecord);',
  10964. 'begin',
  10965. ' if jsvalue(d) then ;',
  10966. ' if jsvalue(c) then ;',
  10967. ' if jsvalue(v) then ;',
  10968. 'end;',
  10969. 'var',
  10970. ' Jv: jsvalue;',
  10971. ' Rec: trecord;',
  10972. 'begin',
  10973. ' rec:=trecord(jv);',
  10974. ' jv:=rec;',
  10975. ' Fly(rec,rec);',
  10976. ' Fly(@rec,@rec);',
  10977. ' if jsvalue(Rec) then ;',
  10978. ' Run(trecord(jv),trecord(jv),rec);',
  10979. '']);
  10980. ConvertProgram;
  10981. CheckSource('TestRecord_JSValue',
  10982. LinesToStr([ // statements
  10983. 'rtl.recNewT(this, "TRecord", function () {',
  10984. ' this.i = 0;',
  10985. ' this.$eq = function (b) {',
  10986. ' return this.i === b.i;',
  10987. ' };',
  10988. ' this.$assign = function (s) {',
  10989. ' this.i = s.i;',
  10990. ' return this;',
  10991. ' };',
  10992. '});',
  10993. 'this.Fly = function (d, c) {',
  10994. '};',
  10995. 'this.Run = function (d, c, v) {',
  10996. ' if (d) ;',
  10997. ' if (c) ;',
  10998. ' if (v) ;',
  10999. '};',
  11000. 'this.Jv = undefined;',
  11001. 'this.Rec = this.TRecord.$new();',
  11002. '']),
  11003. LinesToStr([
  11004. '$mod.Rec.$assign(rtl.getObject($mod.Jv));',
  11005. '$mod.Jv = $mod.Rec;',
  11006. '$mod.Fly($mod.TRecord.$clone($mod.Rec), $mod.Rec);',
  11007. '$mod.Fly($mod.Rec, $mod.Rec);',
  11008. 'if ($mod.Rec) ;',
  11009. '$mod.Run($mod.TRecord.$clone(rtl.getObject($mod.Jv)), rtl.getObject($mod.Jv), $mod.Rec);',
  11010. '']));
  11011. end;
  11012. procedure TTestModule.TestRecord_VariantFail;
  11013. begin
  11014. StartProgram(false);
  11015. Add([
  11016. 'type',
  11017. ' TRec = record',
  11018. ' case word of',
  11019. ' 0: (b0, b1: Byte);',
  11020. ' 1: (i: word);',
  11021. ' end;',
  11022. 'begin']);
  11023. SetExpectedPasResolverError('variant record is not supported',
  11024. nXIsNotSupported);
  11025. ConvertProgram;
  11026. end;
  11027. procedure TTestModule.TestRecord_FieldArray;
  11028. begin
  11029. StartProgram(false);
  11030. Add([
  11031. 'type',
  11032. ' TArrInt = array[3..4] of longint;',
  11033. ' TArrArrInt = array[3..4] of longint;',
  11034. ' TRec = record',
  11035. ' a: array of longint;',
  11036. ' s: array[1..2] of longint;',
  11037. ' m: array[1..2,3..4] of longint;',
  11038. ' o: TArrArrInt;',
  11039. ' end;',
  11040. 'begin']);
  11041. ConvertProgram;
  11042. CheckSource('TestRecord_FieldArray',
  11043. LinesToStr([ // statements
  11044. 'rtl.recNewT(this, "TRec", function () {',
  11045. ' this.$new = function () {',
  11046. ' var r = Object.create(this);',
  11047. ' r.a = [];',
  11048. ' r.s = rtl.arraySetLength(null, 0, 2);',
  11049. ' r.m = rtl.arraySetLength(null, 0, 2, 2);',
  11050. ' r.o = rtl.arraySetLength(null, 0, 2);',
  11051. ' return r;',
  11052. ' };',
  11053. ' this.$eq = function (b) {',
  11054. ' return (this.a === b.a) && rtl.arrayEq(this.s, b.s) && rtl.arrayEq(this.m, b.m) && rtl.arrayEq(this.o, b.o);',
  11055. ' };',
  11056. ' this.$assign = function (s) {',
  11057. ' this.a = rtl.arrayRef(s.a);',
  11058. ' this.s = s.s.slice(0);',
  11059. ' this.m = s.m.slice(0);',
  11060. ' this.o = s.o.slice(0);',
  11061. ' return this;',
  11062. ' };',
  11063. '});',
  11064. '']),
  11065. LinesToStr([ // $mod.$main
  11066. '']));
  11067. end;
  11068. procedure TTestModule.TestRecord_Const;
  11069. begin
  11070. StartProgram(false);
  11071. Add([
  11072. 'type',
  11073. ' TArrInt = array[3..4] of longint;',
  11074. ' TPoint = record x,y: longint; end;',
  11075. ' TRec = record',
  11076. ' i: longint;',
  11077. ' a: array of longint;',
  11078. ' s: array[1..2] of longint;',
  11079. ' m: array[1..2,3..4] of longint;',
  11080. ' p: TPoint;',
  11081. ' end;',
  11082. ' TPoints = array of TPoint;',
  11083. 'const',
  11084. ' r: TRec = (',
  11085. ' i:1;',
  11086. ' a:(2,3);',
  11087. ' s:(4,5);',
  11088. ' m:( (11,12), (13,14) );',
  11089. ' p: (x:21; y:22)',
  11090. ' );',
  11091. ' p: TPoints = ( (x:1;y:2), (x:3;y:4) );',
  11092. 'begin']);
  11093. ConvertProgram;
  11094. CheckSource('TestRecord_Const',
  11095. LinesToStr([ // statements
  11096. 'rtl.recNewT(this, "TPoint", function () {',
  11097. ' this.x = 0;',
  11098. ' this.y = 0;',
  11099. ' this.$eq = function (b) {',
  11100. ' return (this.x === b.x) && (this.y === b.y);',
  11101. ' };',
  11102. ' this.$assign = function (s) {',
  11103. ' this.x = s.x;',
  11104. ' this.y = s.y;',
  11105. ' return this;',
  11106. ' };',
  11107. '});',
  11108. 'rtl.recNewT(this, "TRec", function () {',
  11109. ' this.i = 0;',
  11110. ' this.$new = function () {',
  11111. ' var r = Object.create(this);',
  11112. ' r.a = [];',
  11113. ' r.s = rtl.arraySetLength(null, 0, 2);',
  11114. ' r.m = rtl.arraySetLength(null, 0, 2, 2);',
  11115. ' r.p = $mod.TPoint.$new();',
  11116. ' return r;',
  11117. ' };',
  11118. ' this.$eq = function (b) {',
  11119. ' return (this.i === b.i) && (this.a === b.a) && rtl.arrayEq(this.s, b.s) && rtl.arrayEq(this.m, b.m) && this.p.$eq(b.p);',
  11120. ' };',
  11121. ' this.$assign = function (s) {',
  11122. ' this.i = s.i;',
  11123. ' this.a = rtl.arrayRef(s.a);',
  11124. ' this.s = s.s.slice(0);',
  11125. ' this.m = s.m.slice(0);',
  11126. ' this.p.$assign(s.p);',
  11127. ' return this;',
  11128. ' };',
  11129. '});',
  11130. 'this.r = this.TRec.$clone({',
  11131. ' i: 1,',
  11132. ' a: [2, 3],',
  11133. ' s: [4, 5],',
  11134. ' m: [[11, 12], [13, 14]],',
  11135. ' p: this.TPoint.$clone({',
  11136. ' x: 21,',
  11137. ' y: 22',
  11138. ' })',
  11139. '});',
  11140. 'this.p = [this.TPoint.$clone({',
  11141. ' x: 1,',
  11142. ' y: 2',
  11143. '}), this.TPoint.$clone({',
  11144. ' x: 3,',
  11145. ' y: 4',
  11146. '})];',
  11147. '']),
  11148. LinesToStr([ // $mod.$main
  11149. '']));
  11150. end;
  11151. procedure TTestModule.TestRecord_TypecastFail;
  11152. begin
  11153. StartProgram(false);
  11154. Add([
  11155. 'type',
  11156. ' TPoint = record x,y: longint; end;',
  11157. ' TRec = record l: longint end;',
  11158. 'var p: TPoint;',
  11159. 'begin',
  11160. ' if TRec(p).l=2 then ;']);
  11161. SetExpectedPasResolverError('Illegal type conversion: "TPoint" to "record TRec"',
  11162. nIllegalTypeConversionTo);
  11163. ConvertProgram;
  11164. end;
  11165. procedure TTestModule.TestRecord_InFunction;
  11166. begin
  11167. StartProgram(false);
  11168. Add([
  11169. 'var TPoint: longint = 3;',
  11170. 'procedure DoIt;',
  11171. 'type',
  11172. ' TPoint = record x,y: longint; end;',
  11173. ' TPoints = array of TPoint;',
  11174. 'var',
  11175. ' r: TPoint;',
  11176. ' p: TPoints;',
  11177. 'begin',
  11178. ' SetLength(p,2);',
  11179. 'end;',
  11180. 'begin']);
  11181. ConvertProgram;
  11182. CheckSource('TestRecord_InFunction',
  11183. LinesToStr([ // statements
  11184. 'this.TPoint = 3;',
  11185. 'var TPoint$1 = rtl.recNewT(null, "", function () {',
  11186. ' this.x = 0;',
  11187. ' this.y = 0;',
  11188. ' this.$eq = function (b) {',
  11189. ' return (this.x === b.x) && (this.y === b.y);',
  11190. ' };',
  11191. ' this.$assign = function (s) {',
  11192. ' this.x = s.x;',
  11193. ' this.y = s.y;',
  11194. ' return this;',
  11195. ' };',
  11196. '});',
  11197. 'this.DoIt = function () {',
  11198. ' var r = TPoint$1.$new();',
  11199. ' var p = [];',
  11200. ' p = rtl.arraySetLength(p, TPoint$1, 2);',
  11201. '};',
  11202. '']),
  11203. LinesToStr([ // $mod.$main
  11204. '']));
  11205. end;
  11206. procedure TTestModule.TestRecord_AnonymousFail;
  11207. begin
  11208. StartProgram(false);
  11209. Add([
  11210. 'var',
  11211. ' r: record x: word end;',
  11212. 'begin']);
  11213. SetExpectedPasResolverError('not yet implemented: :TPasRecordType [20190408224556] "anonymous record type"',
  11214. nNotYetImplemented);
  11215. ConvertProgram;
  11216. end;
  11217. procedure TTestModule.TestAdvRecord_Function;
  11218. begin
  11219. StartProgram(false);
  11220. Parser.Options:=Parser.Options+[po_cassignments];
  11221. Add([
  11222. '{$modeswitch AdvancedRecords}',
  11223. 'type',
  11224. ' TPoint = record',
  11225. ' x,y: word;',
  11226. ' function Add(const apt: TPoint): TPoint;',
  11227. ' end;',
  11228. 'function TPoint.Add(const apt: TPoint): TPoint;',
  11229. 'begin',
  11230. ' Result:=Self;',
  11231. ' Result.x+=apt.x;',
  11232. ' Result.y:=Result.y+apt.y;',
  11233. ' Self:=apt;',
  11234. 'end;',
  11235. 'var p,q: TPoint;',
  11236. 'begin',
  11237. ' p.add(q);',
  11238. ' p:=default(TPoint);',
  11239. ' p:=q;',
  11240. '']);
  11241. ConvertProgram;
  11242. CheckSource('TestAdvRecord_Function',
  11243. LinesToStr([ // statements
  11244. 'rtl.recNewT(this, "TPoint", function () {',
  11245. ' this.x = 0;',
  11246. ' this.y = 0;',
  11247. ' this.$eq = function (b) {',
  11248. ' return (this.x === b.x) && (this.y === b.y);',
  11249. ' };',
  11250. ' this.$assign = function (s) {',
  11251. ' this.x = s.x;',
  11252. ' this.y = s.y;',
  11253. ' return this;',
  11254. ' };',
  11255. ' this.Add = function (apt) {',
  11256. ' var Result = $mod.TPoint.$new();',
  11257. ' Result.$assign(this);',
  11258. ' Result.x += apt.x;',
  11259. ' Result.y = Result.y + apt.y;',
  11260. ' this.$assign(apt);',
  11261. ' return Result;',
  11262. ' };',
  11263. '});',
  11264. 'this.p = this.TPoint.$new();',
  11265. 'this.q = this.TPoint.$new();',
  11266. '']),
  11267. LinesToStr([ // $mod.$main
  11268. '$mod.p.Add($mod.q);',
  11269. '$mod.p.$assign($mod.TPoint.$new());',
  11270. '$mod.p.$assign($mod.q);',
  11271. '']));
  11272. end;
  11273. procedure TTestModule.TestAdvRecord_Property;
  11274. begin
  11275. StartProgram(false);
  11276. Add([
  11277. '{$modeswitch AdvancedRecords}',
  11278. 'type',
  11279. ' TPoint = record',
  11280. ' x,y: word;',
  11281. ' strict private',
  11282. ' function GetSize: longword;',
  11283. ' procedure SetSize(Value: longword);',
  11284. ' public',
  11285. ' property Size: longword read GetSize write SetSize;',
  11286. ' property Left: word read x write y;',
  11287. ' end;',
  11288. 'procedure SetSize(Value: longword); begin end;',// check auto rename
  11289. 'function TPoint.GetSize: longword;',
  11290. 'begin',
  11291. ' x:=y;',
  11292. ' Size:=Size;',
  11293. ' Left:=Left;',
  11294. 'end;',
  11295. 'procedure TPoint.SetSize(Value: longword);',
  11296. 'begin',
  11297. 'end;',
  11298. 'var p,q: TPoint;',
  11299. 'begin',
  11300. ' p.Size:=q.Size;',
  11301. ' p.Left:=q.Left;',
  11302. '']);
  11303. ConvertProgram;
  11304. CheckSource('TestAdvRecord_Property',
  11305. LinesToStr([ // statements
  11306. 'rtl.recNewT(this, "TPoint", function () {',
  11307. ' this.x = 0;',
  11308. ' this.y = 0;',
  11309. ' this.$eq = function (b) {',
  11310. ' return (this.x === b.x) && (this.y === b.y);',
  11311. ' };',
  11312. ' this.$assign = function (s) {',
  11313. ' this.x = s.x;',
  11314. ' this.y = s.y;',
  11315. ' return this;',
  11316. ' };',
  11317. ' this.GetSize = function () {',
  11318. ' var Result = 0;',
  11319. ' this.x = this.y;',
  11320. ' this.SetSize(this.GetSize());',
  11321. ' this.y = this.x;',
  11322. ' return Result;',
  11323. ' };',
  11324. ' this.SetSize = function (Value) {',
  11325. ' };',
  11326. '});',
  11327. 'this.SetSize = function (Value) {',
  11328. '};',
  11329. 'this.p = this.TPoint.$new();',
  11330. 'this.q = this.TPoint.$new();',
  11331. '']),
  11332. LinesToStr([ // $mod.$main
  11333. '$mod.p.SetSize($mod.q.GetSize());',
  11334. '$mod.p.y = $mod.q.x;',
  11335. '']));
  11336. end;
  11337. procedure TTestModule.TestAdvRecord_PropertyDefault;
  11338. begin
  11339. StartProgram(false);
  11340. Add([
  11341. '{$modeswitch AdvancedRecords}',
  11342. 'type',
  11343. ' TPoint = record',
  11344. ' strict private',
  11345. ' function GetItems(Index: word): word;',
  11346. ' procedure SetItems(Index: word; Value: word);',
  11347. ' public',
  11348. ' property Items[Index: word]: word read GetItems write SetItems; default;',
  11349. ' end;',
  11350. 'function TPoint.GetItems(Index: word): word;',
  11351. 'begin',
  11352. ' Items[index]:=Items[index];',
  11353. ' self.Items[index]:=self.Items[index];',
  11354. 'end;',
  11355. 'procedure TPoint.SetItems(Index: word; Value: word);',
  11356. 'begin',
  11357. 'end;',
  11358. 'var p: TPoint;',
  11359. 'begin',
  11360. ' p[1]:=p[2];',
  11361. ' p.Items[3]:=p.Items[4];',
  11362. '']);
  11363. ConvertProgram;
  11364. CheckSource('TestAdvRecord_PropertyDefault',
  11365. LinesToStr([ // statements
  11366. 'rtl.recNewT(this, "TPoint", function () {',
  11367. ' this.$eq = function (b) {',
  11368. ' return true;',
  11369. ' };',
  11370. ' this.$assign = function (s) {',
  11371. ' return this;',
  11372. ' };',
  11373. ' this.GetItems = function (Index) {',
  11374. ' var Result = 0;',
  11375. ' this.SetItems(Index, this.GetItems(Index));',
  11376. ' this.SetItems(Index, this.GetItems(Index));',
  11377. ' return Result;',
  11378. ' };',
  11379. ' this.SetItems = function (Index, Value) {',
  11380. ' };',
  11381. '});',
  11382. 'this.p = this.TPoint.$new();',
  11383. '']),
  11384. LinesToStr([ // $mod.$main
  11385. '$mod.p.SetItems(1, $mod.p.GetItems(2));',
  11386. '$mod.p.SetItems(3, $mod.p.GetItems(4));',
  11387. '']));
  11388. end;
  11389. procedure TTestModule.TestAdvRecord_Property_ClassMethod;
  11390. begin
  11391. StartProgram(false);
  11392. Add([
  11393. '{$modeswitch AdvancedRecords}',
  11394. 'type',
  11395. ' TRec = record',
  11396. ' class var',
  11397. ' Fx: longint;',
  11398. ' Fy: longint;',
  11399. ' class function GetInt: longint; static;',
  11400. ' class procedure SetInt(Value: longint); static;',
  11401. ' class procedure DoIt; static;',
  11402. ' class property IntA: longint read Fx write Fy;',
  11403. ' class property IntB: longint read GetInt write SetInt;',
  11404. ' end;',
  11405. 'class function trec.getint: longint;',
  11406. 'begin',
  11407. ' result:=fx;',
  11408. 'end;',
  11409. 'class procedure trec.setint(value: longint);',
  11410. 'begin',
  11411. 'end;',
  11412. 'class procedure trec.doit;',
  11413. 'begin',
  11414. ' IntA:=IntA+1;',
  11415. ' IntB:=IntB+1;',
  11416. 'end;',
  11417. 'var r: trec;',
  11418. 'begin',
  11419. ' trec.inta:=trec.inta+1;',
  11420. ' if trec.intb=2 then;',
  11421. ' trec.intb:=trec.intb+2;',
  11422. ' trec.setint(trec.inta);',
  11423. ' r.inta:=r.inta+1;',
  11424. ' if r.intb=2 then;',
  11425. ' r.intb:=r.intb+2;',
  11426. ' r.setint(r.inta);']);
  11427. ConvertProgram;
  11428. CheckSource('TestAdvRecord_Property_ClassMethod',
  11429. LinesToStr([ // statements
  11430. 'rtl.recNewT(this, "TRec", function () {',
  11431. ' this.Fx = 0;',
  11432. ' this.Fy = 0;',
  11433. ' this.$eq = function (b) {',
  11434. ' return true;',
  11435. ' };',
  11436. ' this.$assign = function (s) {',
  11437. ' return this;',
  11438. ' };',
  11439. ' this.GetInt = function () {',
  11440. ' var Result = 0;',
  11441. ' Result = $mod.TRec.Fx;',
  11442. ' return Result;',
  11443. ' };',
  11444. ' this.SetInt = function (Value) {',
  11445. ' };',
  11446. ' this.DoIt = function () {',
  11447. ' $mod.TRec.Fy = $mod.TRec.Fx + 1;',
  11448. ' $mod.TRec.SetInt($mod.TRec.GetInt() + 1);',
  11449. ' };',
  11450. '}, true);',
  11451. 'this.r = this.TRec.$new();',
  11452. '']),
  11453. LinesToStr([ // $mod.$main
  11454. '$mod.TRec.Fy = $mod.TRec.Fx + 1;',
  11455. 'if ($mod.TRec.GetInt() === 2) ;',
  11456. '$mod.TRec.SetInt($mod.TRec.GetInt() + 2);',
  11457. '$mod.TRec.SetInt($mod.TRec.Fx);',
  11458. '$mod.TRec.Fy = $mod.r.Fx + 1;',
  11459. 'if ($mod.r.GetInt() === 2) ;',
  11460. '$mod.r.SetInt($mod.r.GetInt() + 2);',
  11461. '$mod.r.SetInt($mod.r.Fx);',
  11462. '']));
  11463. end;
  11464. procedure TTestModule.TestAdvRecord_Const;
  11465. begin
  11466. StartProgram(false);
  11467. Add([
  11468. '{$modeswitch AdvancedRecords}',
  11469. 'type',
  11470. ' TArrInt = array[3..4] of longint;',
  11471. ' TPoint = record',
  11472. ' x,y: longint;',
  11473. ' class var Count: nativeint;',
  11474. ' end;',
  11475. ' TRec = record',
  11476. ' i: longint;',
  11477. ' a: array of longint;',
  11478. ' s: array[1..2] of longint;',
  11479. ' m: array[1..2,3..4] of longint;',
  11480. ' p: TPoint;',
  11481. ' end;',
  11482. ' TPoints = array of TPoint;',
  11483. 'const',
  11484. ' r: TRec = (',
  11485. ' i:1;',
  11486. ' a:(2,3);',
  11487. ' s:(4,5);',
  11488. ' m:( (11,12), (13,14) );',
  11489. ' p: (x:21)',
  11490. ' );',
  11491. ' p: TPoints = ( (x:1;y:2), (x:3;y:4) );',
  11492. 'begin']);
  11493. ConvertProgram;
  11494. CheckSource('TestAdvRecord_Const',
  11495. LinesToStr([ // statements
  11496. 'rtl.recNewT(this, "TPoint", function () {',
  11497. ' this.x = 0;',
  11498. ' this.y = 0;',
  11499. ' this.Count = 0;',
  11500. ' this.$eq = function (b) {',
  11501. ' return (this.x === b.x) && (this.y === b.y);',
  11502. ' };',
  11503. ' this.$assign = function (s) {',
  11504. ' this.x = s.x;',
  11505. ' this.y = s.y;',
  11506. ' return this;',
  11507. ' };',
  11508. '}, true);',
  11509. 'rtl.recNewT(this, "TRec", function () {',
  11510. ' this.i = 0;',
  11511. ' this.$new = function () {',
  11512. ' var r = Object.create(this);',
  11513. ' r.a = [];',
  11514. ' r.s = rtl.arraySetLength(null, 0, 2);',
  11515. ' r.m = rtl.arraySetLength(null, 0, 2, 2);',
  11516. ' r.p = $mod.TPoint.$new();',
  11517. ' return r;',
  11518. ' };',
  11519. ' this.$eq = function (b) {',
  11520. ' return (this.i === b.i) && (this.a === b.a) && rtl.arrayEq(this.s, b.s) && rtl.arrayEq(this.m, b.m) && this.p.$eq(b.p);',
  11521. ' };',
  11522. ' this.$assign = function (s) {',
  11523. ' this.i = s.i;',
  11524. ' this.a = rtl.arrayRef(s.a);',
  11525. ' this.s = s.s.slice(0);',
  11526. ' this.m = s.m.slice(0);',
  11527. ' this.p.$assign(s.p);',
  11528. ' return this;',
  11529. ' };',
  11530. '});',
  11531. 'this.r = this.TRec.$clone({',
  11532. ' i: 1,',
  11533. ' a: [2, 3],',
  11534. ' s: [4, 5],',
  11535. ' m: [[11, 12], [13, 14]],',
  11536. ' p: this.TPoint.$clone({',
  11537. ' x: 21,',
  11538. ' y: 0',
  11539. ' })',
  11540. '});',
  11541. 'this.p = [this.TPoint.$clone({',
  11542. ' x: 1,',
  11543. ' y: 2',
  11544. '}), this.TPoint.$clone({',
  11545. ' x: 3,',
  11546. ' y: 4',
  11547. '})];',
  11548. '']),
  11549. LinesToStr([ // $mod.$main
  11550. '']));
  11551. end;
  11552. procedure TTestModule.TestAdvRecord_ExternalField;
  11553. begin
  11554. StartProgram(false);
  11555. Add([
  11556. '{$modeswitch AdvancedRecords}',
  11557. '{$modeswitch externalclass}',
  11558. 'type',
  11559. ' TCar = record',
  11560. ' public',
  11561. ' Intern: longint external name ''$Intern'';',
  11562. ' Intern2: longint external name ''$Intern2'';',
  11563. ' Bracket: longint external name ''["A B"]'';',
  11564. ' procedure DoIt;',
  11565. ' end;',
  11566. 'procedure tcar.doit;',
  11567. 'begin',
  11568. ' Intern:=Intern+1;',
  11569. ' Intern2:=Intern2+2;',
  11570. ' Bracket:=Bracket+3;',
  11571. 'end;',
  11572. 'var Rec: TCar = (intern: 11; intern2: 12; bracket: 13);',
  11573. 'begin',
  11574. ' Rec.intern:=Rec.intern+1;',
  11575. ' Rec.intern2:=Rec.intern2+2;',
  11576. ' Rec.Bracket:=Rec.Bracket+3;',
  11577. ' with Rec do begin',
  11578. ' intern:=intern+1;',
  11579. ' intern2:=intern2+2;',
  11580. ' Bracket:=Bracket+3;',
  11581. ' end;']);
  11582. ConvertProgram;
  11583. CheckSource('TestAdvRecord_ExternalField',
  11584. LinesToStr([ // statements
  11585. 'rtl.recNewT(this, "TCar", function () {',
  11586. ' this.$eq = function (b) {',
  11587. ' return (this.$Intern === b.$Intern) && (this.$Intern2 === b.$Intern2) && (this["A B"] === b["A B"]);',
  11588. ' };',
  11589. ' this.$assign = function (s) {',
  11590. ' this.$Intern = s.$Intern;',
  11591. ' this.$Intern2 = s.$Intern2;',
  11592. ' this["A B"] = s["A B"];',
  11593. ' return this;',
  11594. ' };',
  11595. ' this.DoIt = function () {',
  11596. ' this.$Intern = this.$Intern + 1;',
  11597. ' this.$Intern2 = this.$Intern2 + 2;',
  11598. ' this["A B"] = this["A B"] + 3;',
  11599. ' };',
  11600. '});',
  11601. 'this.Rec = this.TCar.$clone({',
  11602. ' $Intern: 11,',
  11603. ' $Intern2: 12,',
  11604. ' "A B": 13',
  11605. '});',
  11606. '']),
  11607. LinesToStr([ // $mod.$main
  11608. '$mod.Rec.$Intern = $mod.Rec.$Intern + 1;',
  11609. '$mod.Rec.$Intern2 = $mod.Rec.$Intern2 + 2;',
  11610. '$mod.Rec["A B"] = $mod.Rec["A B"] + 3;',
  11611. 'var $with = $mod.Rec;',
  11612. '$with.$Intern = $with.$Intern + 1;',
  11613. '$with.$Intern2 = $with.$Intern2 + 2;',
  11614. '$with["A B"] = $with["A B"] + 3;',
  11615. '']));
  11616. end;
  11617. procedure TTestModule.TestAdvRecord_SubRecord;
  11618. begin
  11619. StartProgram(false);
  11620. Add([
  11621. '{$modeswitch AdvancedRecords}',
  11622. 'type',
  11623. ' TRec = record',
  11624. ' type',
  11625. ' TPoint = record',
  11626. ' x,y: longint;',
  11627. ' class var Count: nativeint;',
  11628. ' procedure DoIt;',
  11629. ' class procedure DoThat; static;',
  11630. ' end;',
  11631. ' var',
  11632. ' i: longint;',
  11633. ' p: TPoint;',
  11634. ' procedure DoSome;',
  11635. ' end;',
  11636. 'const',
  11637. ' r: TRec = (',
  11638. ' i:1;',
  11639. ' p: (x:21;y:22)',
  11640. ' );',
  11641. 'procedure TRec.DoSome;',
  11642. 'begin',
  11643. ' p.x:=p.y+1;',
  11644. ' p.Count:=p.Count+2;',
  11645. 'end;',
  11646. 'procedure TRec.TPoint.DoIt;',
  11647. 'begin',
  11648. ' Count:=Count+3;',
  11649. 'end;',
  11650. 'class procedure TRec.TPoint.DoThat;',
  11651. 'begin',
  11652. ' Count:=Count+4;',
  11653. 'end;',
  11654. 'begin']);
  11655. ConvertProgram;
  11656. CheckSource('TestAdvRecord_SubRecord',
  11657. LinesToStr([ // statements
  11658. 'rtl.recNewT(this, "TRec", function () {',
  11659. ' rtl.recNewT(this, "TPoint", function () {',
  11660. ' this.x = 0;',
  11661. ' this.y = 0;',
  11662. ' this.Count = 0;',
  11663. ' this.$eq = function (b) {',
  11664. ' return (this.x === b.x) && (this.y === b.y);',
  11665. ' };',
  11666. ' this.$assign = function (s) {',
  11667. ' this.x = s.x;',
  11668. ' this.y = s.y;',
  11669. ' return this;',
  11670. ' };',
  11671. ' this.DoIt = function () {',
  11672. ' $mod.TRec.TPoint.Count = this.Count + 3;',
  11673. ' };',
  11674. ' this.DoThat = function () {',
  11675. ' $mod.TRec.TPoint.Count = $mod.TRec.TPoint.Count + 4;',
  11676. ' };',
  11677. ' }, true);',
  11678. ' this.i = 0;',
  11679. ' this.$new = function () {',
  11680. ' var r = Object.create(this);',
  11681. ' r.p = this.TPoint.$new();',
  11682. ' return r;',
  11683. ' };',
  11684. ' this.$eq = function (b) {',
  11685. ' return (this.i === b.i) && this.p.$eq(b.p);',
  11686. ' };',
  11687. ' this.$assign = function (s) {',
  11688. ' this.i = s.i;',
  11689. ' this.p.$assign(s.p);',
  11690. ' return this;',
  11691. ' };',
  11692. ' this.DoSome = function () {',
  11693. ' this.p.x = this.p.y + 1;',
  11694. ' this.TPoint.Count = this.p.Count + 2;',
  11695. ' };',
  11696. '}, true);',
  11697. 'this.r = this.TRec.$clone({',
  11698. ' i: 1,',
  11699. ' p: this.TRec.TPoint.$clone({',
  11700. ' x: 21,',
  11701. ' y: 22',
  11702. ' })',
  11703. '});',
  11704. '']),
  11705. LinesToStr([ // $mod.$main
  11706. '']));
  11707. end;
  11708. procedure TTestModule.TestAdvRecord_SubClass;
  11709. begin
  11710. StartProgram(false);
  11711. Add([
  11712. '{$modeswitch AdvancedRecords}',
  11713. 'type',
  11714. ' TObject = class end;',
  11715. ' TPoint = record',
  11716. ' type',
  11717. ' TBird = class',
  11718. ' procedure DoIt;',
  11719. ' class procedure Glob;',
  11720. ' end;',
  11721. ' procedure DoIt(b: TBird);',
  11722. ' end;',
  11723. 'procedure TPoint.TBird.DoIt;',
  11724. 'begin',
  11725. ' doit;',
  11726. ' self.doit;',
  11727. ' glob;',
  11728. ' self.glob;',
  11729. 'end;',
  11730. 'class procedure TPoint.TBird.Glob;',
  11731. 'begin',
  11732. ' glob;',
  11733. ' self.glob;',
  11734. 'end;',
  11735. 'procedure TPoint.DoIt(b: TBird);',
  11736. 'begin',
  11737. ' b.doit;',
  11738. ' b.glob;',
  11739. ' TBird.glob;',
  11740. 'end;',
  11741. 'begin',
  11742. '']);
  11743. ConvertProgram;
  11744. CheckSource('TestAdvRecord_SubClass',
  11745. LinesToStr([ // statements
  11746. 'rtl.createClass(this, "TObject", null, function () {',
  11747. ' this.$init = function () {',
  11748. ' };',
  11749. ' this.$final = function () {',
  11750. ' };',
  11751. '});',
  11752. 'rtl.recNewT(this, "TPoint", function () {',
  11753. ' rtl.createClass(this, "TBird", this.TObject, function () {',
  11754. ' this.DoIt = function () {',
  11755. ' this.DoIt();',
  11756. ' this.DoIt();',
  11757. ' this.$class.Glob();',
  11758. ' this.$class.Glob();',
  11759. ' };',
  11760. ' this.Glob = function () {',
  11761. ' this.Glob();',
  11762. ' this.Glob();',
  11763. ' };',
  11764. ' }, "TPoint.TBird");',
  11765. ' this.$eq = function (b) {',
  11766. ' return true;',
  11767. ' };',
  11768. ' this.$assign = function (s) {',
  11769. ' return this;',
  11770. ' };',
  11771. ' this.DoIt = function (b) {',
  11772. ' b.DoIt();',
  11773. ' b.$class.Glob();',
  11774. ' this.TBird.Glob();',
  11775. ' };',
  11776. '}, true);',
  11777. '']),
  11778. LinesToStr([ // $mod.$main
  11779. '']));
  11780. end;
  11781. procedure TTestModule.TestAdvRecord_SubInterfaceFail;
  11782. begin
  11783. StartProgram(false);
  11784. Add([
  11785. '{$modeswitch AdvancedRecords}',
  11786. 'type',
  11787. ' IUnknown = interface end;',
  11788. ' TPoint = record',
  11789. ' type IBird = interface end;',
  11790. ' end;',
  11791. 'begin',
  11792. '']);
  11793. SetExpectedPasResolverError('not yet implemented: IBird:TPasClassType [20190105143752] "interface inside record"',
  11794. nNotYetImplemented);
  11795. ParseProgram;
  11796. end;
  11797. procedure TTestModule.TestAdvRecord_Constructor;
  11798. begin
  11799. StartProgram(false);
  11800. Add([
  11801. '{$modeswitch AdvancedRecords}',
  11802. 'type',
  11803. ' TPoint = record',
  11804. ' x,y: longint;',
  11805. ' constructor Create(ax: longint; ay: longint = -1);',
  11806. ' end;',
  11807. 'constructor tpoint.create(ax,ay: longint);',
  11808. 'begin',
  11809. ' x:=ax;',
  11810. ' self.y:=ay;',
  11811. 'end;',
  11812. 'var r: TPoint;',
  11813. 'begin',
  11814. ' r:=TPoint.Create(1,2);',
  11815. ' with TPoint do r:=Create(1,2);',
  11816. ' r.Create(3);',
  11817. ' r:=r.Create(4);',
  11818. '']);
  11819. ConvertProgram;
  11820. CheckSource('TestAdvRecord_Constructor',
  11821. LinesToStr([ // statements
  11822. 'rtl.recNewT(this, "TPoint", function () {',
  11823. ' this.x = 0;',
  11824. ' this.y = 0;',
  11825. ' this.$eq = function (b) {',
  11826. ' return (this.x === b.x) && (this.y === b.y);',
  11827. ' };',
  11828. ' this.$assign = function (s) {',
  11829. ' this.x = s.x;',
  11830. ' this.y = s.y;',
  11831. ' return this;',
  11832. ' };',
  11833. ' this.Create = function (ax, ay) {',
  11834. ' this.x = ax;',
  11835. ' this.y = ay;',
  11836. ' return this;',
  11837. ' };',
  11838. '}, true);',
  11839. 'this.r = this.TPoint.$new();',
  11840. '']),
  11841. LinesToStr([ // $mod.$main
  11842. '$mod.r.$assign($mod.TPoint.$new().Create(1, 2));',
  11843. 'var $with = $mod.TPoint;',
  11844. '$mod.r.$assign($with.$new().Create(1, 2));',
  11845. '$mod.r.Create(3, -1);',
  11846. '$mod.r.$assign($mod.r.Create(4, -1));',
  11847. '']));
  11848. end;
  11849. procedure TTestModule.TestAdvRecord_ClassConstructor_Program;
  11850. begin
  11851. StartProgram(false);
  11852. Add([
  11853. '{$modeswitch AdvancedRecords}',
  11854. 'type',
  11855. ' TPoint = record',
  11856. ' class var x: longint;',
  11857. ' class procedure Fly; static;',
  11858. ' class constructor Init;',
  11859. ' end;',
  11860. 'var count: word;',
  11861. 'class procedure Tpoint.Fly;',
  11862. 'begin',
  11863. 'end;',
  11864. 'class constructor tpoint.init;',
  11865. 'begin',
  11866. ' count:=count+1;',
  11867. ' x:=x+3;',
  11868. ' tpoint.x:=tpoint.x+4;',
  11869. ' fly;',
  11870. ' tpoint.fly;',
  11871. 'end;',
  11872. 'var r: TPoint;',
  11873. 'begin',
  11874. ' r.x:=r.x+10;',
  11875. ' r.Fly;',
  11876. ' r.Fly();',
  11877. '']);
  11878. ConvertProgram;
  11879. CheckSource('TestAdvRecord_ClassConstructor_Program',
  11880. LinesToStr([ // statements
  11881. 'rtl.recNewT(this, "TPoint", function () {',
  11882. ' this.x = 0;',
  11883. ' this.$eq = function (b) {',
  11884. ' return true;',
  11885. ' };',
  11886. ' this.$assign = function (s) {',
  11887. ' return this;',
  11888. ' };',
  11889. ' this.Fly = function () {',
  11890. ' };',
  11891. '}, true);',
  11892. 'this.count = 0;',
  11893. 'this.r = this.TPoint.$new();',
  11894. '']),
  11895. LinesToStr([ // $mod.$main
  11896. '(function () {',
  11897. ' $mod.count = $mod.count + 1;',
  11898. ' $mod.TPoint.x = $mod.TPoint.x + 3;',
  11899. ' $mod.TPoint.x = $mod.TPoint.x + 4;',
  11900. ' $mod.TPoint.Fly();',
  11901. ' $mod.TPoint.Fly();',
  11902. '})();',
  11903. '$mod.TPoint.x = $mod.r.x + 10;',
  11904. '$mod.r.Fly();',
  11905. '$mod.r.Fly();',
  11906. '']));
  11907. end;
  11908. procedure TTestModule.TestAdvRecord_ClassConstructor_Unit;
  11909. begin
  11910. StartUnit(false);
  11911. Add([
  11912. 'interface',
  11913. '{$modeswitch AdvancedRecords}',
  11914. 'type',
  11915. ' TPoint = record',
  11916. ' class var x: longint;',
  11917. ' class procedure Fly; static;',
  11918. ' class constructor Init;',
  11919. ' end;',
  11920. 'implementation',
  11921. 'var count: word;',
  11922. 'class procedure Tpoint.Fly;',
  11923. 'begin',
  11924. 'end;',
  11925. 'class constructor tpoint.init;',
  11926. 'begin',
  11927. ' count:=count+1;',
  11928. ' x:=3;',
  11929. ' tpoint.x:=4;',
  11930. ' fly;',
  11931. ' tpoint.fly;',
  11932. 'end;',
  11933. '']);
  11934. ConvertUnit;
  11935. CheckSource('TestAdvRecord_ClassConstructor_Unit',
  11936. LinesToStr([ // statements
  11937. 'var $impl = $mod.$impl;',
  11938. 'rtl.recNewT(this, "TPoint", function () {',
  11939. ' this.x = 0;',
  11940. ' this.$eq = function (b) {',
  11941. ' return true;',
  11942. ' };',
  11943. ' this.$assign = function (s) {',
  11944. ' return this;',
  11945. ' };',
  11946. ' this.Fly = function () {',
  11947. ' };',
  11948. '}, true);',
  11949. '']),
  11950. LinesToStr([ // $mod.$init
  11951. '(function () {',
  11952. ' $impl.count = $impl.count + 1;',
  11953. ' $mod.TPoint.x = 3;',
  11954. ' $mod.TPoint.x = 4;',
  11955. ' $mod.TPoint.Fly();',
  11956. ' $mod.TPoint.Fly();',
  11957. '})();',
  11958. '']),
  11959. LinesToStr([ // $mod.$main
  11960. '$impl.count = 0;',
  11961. '']));
  11962. end;
  11963. procedure TTestModule.TestClass_TObjectDefaultConstructor;
  11964. begin
  11965. StartProgram(false);
  11966. Add(['type',
  11967. ' TObject = class',
  11968. ' public',
  11969. ' constructor Create;',
  11970. ' destructor Destroy;',
  11971. ' end;',
  11972. ' TBird = TObject;',
  11973. 'constructor tobject.create;',
  11974. 'begin end;',
  11975. 'destructor tobject.destroy;',
  11976. 'begin end;',
  11977. 'var Obj: tobject;',
  11978. 'begin',
  11979. ' obj:=tobject.create;',
  11980. ' obj:=tobject.create();',
  11981. ' obj:=tbird.create;',
  11982. ' obj:=tbird.create();',
  11983. ' obj:=obj.create();',
  11984. ' obj.destroy;',
  11985. '']);
  11986. ConvertProgram;
  11987. CheckSource('TestClass_TObjectDefaultConstructor',
  11988. LinesToStr([ // statements
  11989. 'rtl.createClass(this,"TObject",null,function(){',
  11990. ' this.$init = function () {',
  11991. ' };',
  11992. ' this.$final = function () {',
  11993. ' };',
  11994. ' this.Create = function(){',
  11995. ' return this;',
  11996. ' };',
  11997. ' this.Destroy = function(){',
  11998. ' };',
  11999. '});',
  12000. 'this.Obj = null;'
  12001. ]),
  12002. LinesToStr([ // $mod.$main
  12003. '$mod.Obj = $mod.TObject.$create("Create");',
  12004. '$mod.Obj = $mod.TObject.$create("Create");',
  12005. '$mod.Obj = $mod.TObject.$create("Create");',
  12006. '$mod.Obj = $mod.TObject.$create("Create");',
  12007. '$mod.Obj = $mod.Obj.Create();',
  12008. '$mod.Obj.$destroy("Destroy");',
  12009. '']));
  12010. end;
  12011. procedure TTestModule.TestClass_TObjectConstructorWithParams;
  12012. begin
  12013. StartProgram(false);
  12014. Add('type');
  12015. Add(' TObject = class');
  12016. Add(' public');
  12017. Add(' constructor Create(Par: longint);');
  12018. Add(' end;');
  12019. Add('constructor tobject.create(par: longint);');
  12020. Add('begin end;');
  12021. Add('var Obj: tobject;');
  12022. Add('begin');
  12023. Add(' obj:=tobject.create(3);');
  12024. ConvertProgram;
  12025. CheckSource('TestClass_TObjectConstructorWithParams',
  12026. LinesToStr([ // statements
  12027. 'rtl.createClass(this,"TObject",null,function(){',
  12028. ' this.$init = function () {',
  12029. ' };',
  12030. ' this.$final = function () {',
  12031. ' };',
  12032. ' this.Create = function(Par){',
  12033. ' return this;',
  12034. ' };',
  12035. '});',
  12036. 'this.Obj = null;'
  12037. ]),
  12038. LinesToStr([ // $mod.$main
  12039. '$mod.Obj = $mod.TObject.$create("Create",[3]);'
  12040. ]));
  12041. end;
  12042. procedure TTestModule.TestClass_TObjectConstructorWithDefaultParam;
  12043. begin
  12044. StartProgram(false);
  12045. Add('type');
  12046. Add(' TObject = class');
  12047. Add(' public');
  12048. Add(' constructor Create;');
  12049. Add(' end;');
  12050. Add(' TTest = class(TObject)');
  12051. Add(' public');
  12052. Add(' constructor Create(const Par: longint = 1);');
  12053. Add(' end;');
  12054. Add('constructor tobject.create;');
  12055. Add('begin end;');
  12056. Add('constructor ttest.create(const par: longint);');
  12057. Add('begin end;');
  12058. Add('var t: ttest;');
  12059. Add('begin');
  12060. Add(' t:=ttest.create;');
  12061. Add(' t:=ttest.create(2);');
  12062. ConvertProgram;
  12063. CheckSource('TestClass_TObjectConstructorWithDefaultParam',
  12064. LinesToStr([ // statements
  12065. 'rtl.createClass(this,"TObject",null,function(){',
  12066. ' this.$init = function () {',
  12067. ' };',
  12068. ' this.$final = function () {',
  12069. ' };',
  12070. ' this.Create = function(){',
  12071. ' return this;',
  12072. ' };',
  12073. '});',
  12074. 'rtl.createClass(this, "TTest", this.TObject, function () {',
  12075. ' this.Create$1 = function (Par) {',
  12076. ' return this;',
  12077. ' };',
  12078. '});',
  12079. 'this.t = null;'
  12080. ]),
  12081. LinesToStr([ // $mod.$main
  12082. '$mod.t = $mod.TTest.$create("Create$1", [1]);',
  12083. '$mod.t = $mod.TTest.$create("Create$1", [2]);'
  12084. ]));
  12085. end;
  12086. procedure TTestModule.TestClass_Var;
  12087. begin
  12088. StartProgram(false);
  12089. Add([
  12090. 'type',
  12091. ' TObject = class',
  12092. ' public',
  12093. ' vI: longint;',
  12094. ' constructor Create(Par: longint);',
  12095. ' end;',
  12096. 'constructor tobject.create(par: longint);',
  12097. 'begin',
  12098. ' vi:=par+3',
  12099. 'end;',
  12100. 'var Obj: tobject;',
  12101. 'begin',
  12102. ' obj:=tobject.create(4);',
  12103. ' obj.vi:=obj.VI+5;']);
  12104. ConvertProgram;
  12105. CheckSource('TestClass_Var',
  12106. LinesToStr([ // statements
  12107. 'rtl.createClass(this,"TObject",null,function(){',
  12108. ' this.$init = function () {',
  12109. ' this.vI = 0;',
  12110. ' };',
  12111. ' this.$final = function () {',
  12112. ' };',
  12113. ' this.Create = function(Par){',
  12114. ' this.vI = Par+3;',
  12115. ' return this;',
  12116. ' };',
  12117. '});',
  12118. 'this.Obj = null;'
  12119. ]),
  12120. LinesToStr([ // $mod.$main
  12121. '$mod.Obj = $mod.TObject.$create("Create",[4]);',
  12122. '$mod.Obj.vI = $mod.Obj.vI + 5;'
  12123. ]));
  12124. end;
  12125. procedure TTestModule.TestClass_Method;
  12126. begin
  12127. StartProgram(false);
  12128. Add('type');
  12129. Add(' TObject = class');
  12130. Add(' public');
  12131. Add(' vI: longint;');
  12132. Add(' Sub: TObject;');
  12133. Add(' constructor Create;');
  12134. Add(' function GetIt(Par: longint): tobject;');
  12135. Add(' end;');
  12136. Add('constructor tobject.create; begin end;');
  12137. Add('function tobject.getit(par: longint): tobject;');
  12138. Add('begin');
  12139. Add(' Self.vi:=par+3;');
  12140. Add(' Result:=self.sub;');
  12141. Add('end;');
  12142. Add('var Obj: tobject;');
  12143. Add('begin');
  12144. Add(' obj:=tobject.create;');
  12145. Add(' obj.getit(4);');
  12146. Add(' obj.sub.sub:=nil;');
  12147. Add(' obj.sub.getit(5);');
  12148. Add(' obj.sub.getit(6).SUB:=nil;');
  12149. Add(' obj.sub.getit(7).GETIT(8);');
  12150. Add(' obj.sub.getit(9).SuB.getit(10);');
  12151. ConvertProgram;
  12152. CheckSource('TestClass_Method',
  12153. LinesToStr([ // statements
  12154. 'rtl.createClass(this,"TObject",null,function(){',
  12155. ' this.$init = function () {',
  12156. ' this.vI = 0;',
  12157. ' this.Sub = null;',
  12158. ' };',
  12159. ' this.$final = function () {',
  12160. ' this.Sub = undefined;',
  12161. ' };',
  12162. ' this.Create = function(){',
  12163. ' return this;',
  12164. ' };',
  12165. ' this.GetIt = function(Par){',
  12166. ' var Result = null;',
  12167. ' this.vI = Par + 3;',
  12168. ' Result = this.Sub;',
  12169. ' return Result;',
  12170. ' };',
  12171. '});',
  12172. 'this.Obj = null;'
  12173. ]),
  12174. LinesToStr([ // $mod.$main
  12175. '$mod.Obj = $mod.TObject.$create("Create");',
  12176. '$mod.Obj.GetIt(4);',
  12177. '$mod.Obj.Sub.Sub=null;',
  12178. '$mod.Obj.Sub.GetIt(5);',
  12179. '$mod.Obj.Sub.GetIt(6).Sub=null;',
  12180. '$mod.Obj.Sub.GetIt(7).GetIt(8);',
  12181. '$mod.Obj.Sub.GetIt(9).Sub.GetIt(10);'
  12182. ]));
  12183. end;
  12184. procedure TTestModule.TestClass_Implementation;
  12185. begin
  12186. StartUnit(false);
  12187. Add([
  12188. 'interface',
  12189. 'type',
  12190. ' TObject = class',
  12191. ' constructor Create;',
  12192. ' end;',
  12193. 'implementation',
  12194. 'type',
  12195. ' TIntClass = class',
  12196. ' constructor Create; reintroduce;',
  12197. ' class procedure DoGlob;',
  12198. ' end;',
  12199. 'constructor tintclass.create;',
  12200. 'begin',
  12201. ' inherited;',
  12202. ' inherited create;',
  12203. ' doglob;',
  12204. 'end;',
  12205. 'class procedure tintclass.doglob;',
  12206. 'begin',
  12207. 'end;',
  12208. 'constructor tobject.create;',
  12209. 'var',
  12210. ' iC: tintclass;',
  12211. 'begin',
  12212. ' ic:=tintclass.create;',
  12213. ' tintclass.doglob;',
  12214. ' ic.doglob;',
  12215. 'end;',
  12216. 'initialization',
  12217. ' tintclass.doglob;',
  12218. '']);
  12219. ConvertUnit;
  12220. CheckSource('TestClass_Implementation',
  12221. LinesToStr([ // statements
  12222. 'var $impl = $mod.$impl;',
  12223. 'rtl.createClass(this, "TObject", null, function () {',
  12224. ' this.$init = function () {',
  12225. ' };',
  12226. ' this.$final = function () {',
  12227. ' };',
  12228. ' this.Create = function () {',
  12229. ' var iC = null;',
  12230. ' iC = $impl.TIntClass.$create("Create$1");',
  12231. ' $impl.TIntClass.DoGlob();',
  12232. ' iC.$class.DoGlob();',
  12233. ' return this;',
  12234. ' };',
  12235. '});',
  12236. '']),
  12237. LinesToStr([ // $mod.$main
  12238. '$impl.TIntClass.DoGlob();',
  12239. '']),
  12240. LinesToStr([
  12241. 'rtl.createClass($impl, "TIntClass", $mod.TObject, function () {',
  12242. ' this.Create$1 = function () {',
  12243. ' $mod.TObject.Create.call(this);',
  12244. ' $mod.TObject.Create.call(this);',
  12245. ' this.$class.DoGlob();',
  12246. ' return this;',
  12247. ' };',
  12248. ' this.DoGlob = function () {',
  12249. ' };',
  12250. '});',
  12251. '']));
  12252. end;
  12253. procedure TTestModule.TestClass_Inheritance;
  12254. begin
  12255. StartProgram(false);
  12256. Add('type');
  12257. Add(' TObject = class');
  12258. Add(' public');
  12259. Add(' constructor Create;');
  12260. Add(' end;');
  12261. Add(' TClassA = class');
  12262. Add(' end;');
  12263. Add(' TClassB = class(TObject)');
  12264. Add(' procedure ProcB;');
  12265. Add(' end;');
  12266. Add('constructor tobject.create; begin end;');
  12267. Add('procedure tclassb.procb; begin end;');
  12268. Add('var');
  12269. Add(' oO: TObject;');
  12270. Add(' oA: TClassA;');
  12271. Add(' oB: TClassB;');
  12272. Add('begin');
  12273. Add(' oO:=tobject.Create;');
  12274. Add(' oA:=tclassa.Create;');
  12275. Add(' ob:=tclassb.Create;');
  12276. Add(' if oo is tclassa then ;');
  12277. Add(' ob:=oo as tclassb;');
  12278. Add(' (oo as tclassb).procb;');
  12279. ConvertProgram;
  12280. CheckSource('TestClass_Inheritance',
  12281. LinesToStr([ // statements
  12282. 'rtl.createClass(this,"TObject",null,function(){',
  12283. ' this.$init = function () {',
  12284. ' };',
  12285. ' this.$final = function () {',
  12286. ' };',
  12287. ' this.Create = function () {',
  12288. ' return this;',
  12289. ' };',
  12290. '});',
  12291. 'rtl.createClass(this,"TClassA",this.TObject,function(){',
  12292. '});',
  12293. 'rtl.createClass(this,"TClassB",this.TObject,function(){',
  12294. ' this.ProcB = function () {',
  12295. ' };',
  12296. '});',
  12297. 'this.oO = null;',
  12298. 'this.oA = null;',
  12299. 'this.oB = null;'
  12300. ]),
  12301. LinesToStr([ // $mod.$main
  12302. '$mod.oO = $mod.TObject.$create("Create");',
  12303. '$mod.oA = $mod.TClassA.$create("Create");',
  12304. '$mod.oB = $mod.TClassB.$create("Create");',
  12305. 'if ($mod.TClassA.isPrototypeOf($mod.oO));',
  12306. '$mod.oB = rtl.as($mod.oO, $mod.TClassB);',
  12307. 'rtl.as($mod.oO, $mod.TClassB).ProcB();'
  12308. ]));
  12309. end;
  12310. procedure TTestModule.TestClass_TypeAlias;
  12311. begin
  12312. StartProgram(false);
  12313. Add([
  12314. '{$interfaces corba}',
  12315. 'type',
  12316. ' IObject = interface',
  12317. ' end;',
  12318. ' IBird = type IObject;',
  12319. ' TObject = class',
  12320. ' end;',
  12321. ' TBird = type TObject;',
  12322. 'var',
  12323. ' oObj: TObject;',
  12324. ' oBird: TBird;',
  12325. ' IntfObj: IObject;',
  12326. ' IntfBird: IBird;',
  12327. 'begin',
  12328. ' oObj:=oBird;',
  12329. '']);
  12330. ConvertProgram;
  12331. CheckSource('TestClass_TypeAlias',
  12332. LinesToStr([ // statements
  12333. 'rtl.createInterface(this, "IObject", "{B92D5841-6F2A-306A-8000-000000000000}", [], null);',
  12334. 'rtl.createInterface(this, "IBird", "{4B0D080B-C0F6-387B-AE88-F10981585074}", [], this.IObject);',
  12335. 'rtl.createClass(this, "TObject", null, function () {',
  12336. ' this.$init = function () {',
  12337. ' };',
  12338. ' this.$final = function () {',
  12339. ' };',
  12340. '});',
  12341. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  12342. '});',
  12343. 'this.oObj = null;',
  12344. 'this.oBird = null;',
  12345. 'this.IntfObj = null;',
  12346. 'this.IntfBird = null;',
  12347. '']),
  12348. LinesToStr([ // $mod.$main
  12349. '$mod.oObj = $mod.oBird;',
  12350. '']));
  12351. end;
  12352. procedure TTestModule.TestClass_AbstractMethod;
  12353. begin
  12354. StartProgram(false);
  12355. Add('type');
  12356. Add(' TObject = class');
  12357. Add(' public');
  12358. Add(' procedure DoIt; virtual; abstract;');
  12359. Add(' end;');
  12360. Add('begin');
  12361. ConvertProgram;
  12362. CheckSource('TestClass_AbstractMethod',
  12363. LinesToStr([ // statements
  12364. 'rtl.createClass(this,"TObject",null,function(){',
  12365. ' this.$init = function () {',
  12366. ' };',
  12367. ' this.$final = function () {',
  12368. ' };',
  12369. '});'
  12370. ]),
  12371. LinesToStr([ // this.$main
  12372. ''
  12373. ]));
  12374. end;
  12375. procedure TTestModule.TestClass_CallInherited_ProcNoParams;
  12376. begin
  12377. StartProgram(false);
  12378. Add([
  12379. 'type',
  12380. ' TObject = class',
  12381. ' procedure DoAbstract; virtual; abstract;',
  12382. ' procedure DoVirtual; virtual;',
  12383. ' procedure DoIt;',
  12384. ' end;',
  12385. ' TA = class',
  12386. ' procedure doabstract; override;',
  12387. ' procedure dovirtual; override;',
  12388. ' procedure DoSome;',
  12389. ' end;',
  12390. 'procedure tobject.dovirtual;',
  12391. 'begin',
  12392. ' inherited; // call non existing ancestor -> ignore silently',
  12393. 'end;',
  12394. 'procedure tobject.doit;',
  12395. 'begin',
  12396. 'end;',
  12397. 'procedure ta.doabstract;',
  12398. 'begin',
  12399. ' inherited dovirtual; // call TObject.DoVirtual',
  12400. 'end;',
  12401. 'procedure ta.dovirtual;',
  12402. 'begin',
  12403. ' inherited; // call TObject.DoVirtual',
  12404. ' inherited dovirtual; // call TObject.DoVirtual',
  12405. ' inherited dovirtual(); // call TObject.DoVirtual',
  12406. ' doit;',
  12407. ' doit();',
  12408. 'end;',
  12409. 'procedure ta.dosome;',
  12410. 'begin',
  12411. ' inherited; // call non existing ancestor method -> silently ignore',
  12412. 'end;',
  12413. 'begin']);
  12414. ConvertProgram;
  12415. CheckSource('TestClass_CallInherited_ProcNoParams',
  12416. LinesToStr([ // statements
  12417. 'rtl.createClass(this,"TObject",null,function(){',
  12418. ' this.$init = function () {',
  12419. ' };',
  12420. ' this.$final = function () {',
  12421. ' };',
  12422. ' this.DoVirtual = function () {',
  12423. ' };',
  12424. ' this.DoIt = function () {',
  12425. ' };',
  12426. '});',
  12427. 'rtl.createClass(this, "TA", this.TObject, function () {',
  12428. ' this.DoAbstract = function () {',
  12429. ' $mod.TObject.DoVirtual.call(this);',
  12430. ' };',
  12431. ' this.DoVirtual = function () {',
  12432. ' $mod.TObject.DoVirtual.call(this);',
  12433. ' $mod.TObject.DoVirtual.call(this);',
  12434. ' $mod.TObject.DoVirtual.call(this);',
  12435. ' this.DoIt();',
  12436. ' this.DoIt();',
  12437. ' };',
  12438. ' this.DoSome = function () {',
  12439. ' };',
  12440. '});'
  12441. ]),
  12442. LinesToStr([ // this.$main
  12443. ''
  12444. ]));
  12445. end;
  12446. procedure TTestModule.TestClass_CallInherited_WithParams;
  12447. begin
  12448. StartProgram(false);
  12449. Add([
  12450. 'type',
  12451. ' TObject = class',
  12452. ' procedure DoAbstract(pA: longint; pB: longint = 0); virtual; abstract;',
  12453. ' procedure DoVirtual(pA: longint; pB: longint = 0); virtual;',
  12454. ' procedure DoIt(pA: longint; pB: longint = 0);',
  12455. ' procedure DoIt2(pA: longint = 1; pB: longint = 2);',
  12456. ' function GetIt(pA: longint = 1; pB: longint = 2): longint;',
  12457. ' end;',
  12458. ' TClassA = class',
  12459. ' procedure DoAbstract(pA: longint; pB: longint = 0); override;',
  12460. ' procedure DoVirtual(pA: longint; pB: longint = 0); override;',
  12461. ' function GetIt(pA: longint = 1; pB: longint = 2): longint;',
  12462. ' end;',
  12463. 'procedure tobject.dovirtual(pa: longint; pb: longint = 0);',
  12464. 'begin',
  12465. 'end;',
  12466. 'procedure tobject.doit(pa: longint; pb: longint = 0);',
  12467. 'begin',
  12468. 'end;',
  12469. 'procedure tobject.doit2(pa: longint; pb: longint = 0);',
  12470. 'begin',
  12471. 'end;',
  12472. 'function tobject.getit(pa: longint; pb: longint = 0): longint;',
  12473. 'begin',
  12474. 'end;',
  12475. 'procedure tclassa.doabstract(pa: longint; pb: longint = 0);',
  12476. 'begin',
  12477. ' inherited dovirtual(pa,pb); // call TObject.DoVirtual(pA,pB)',
  12478. ' inherited dovirtual(pa); // call TObject.DoVirtual(pA,0)',
  12479. 'end;',
  12480. 'procedure tclassa.dovirtual(pa: longint; pb: longint = 0);',
  12481. 'begin',
  12482. ' inherited; // call TObject.DoVirtual(pA,pB)',
  12483. ' inherited dovirtual(pa,pb); // call TObject.DoVirtual(pA,pB)',
  12484. ' inherited dovirtual(pa); // call TObject.DoVirtual(pA,0)',
  12485. ' doit(pa,pb);',
  12486. ' doit(pa);',
  12487. ' doit2(pa);',
  12488. ' doit2;',
  12489. 'end;',
  12490. 'function tclassa.getit(pa: longint; pb: longint = 0): longint;',
  12491. 'begin',
  12492. ' pa:=inherited;',
  12493. 'end;',
  12494. 'begin']);
  12495. ConvertProgram;
  12496. CheckSource('TestClass_CallInherited_WithParams',
  12497. LinesToStr([ // statements
  12498. 'rtl.createClass(this,"TObject",null,function(){',
  12499. ' this.$init = function () {',
  12500. ' };',
  12501. ' this.$final = function () {',
  12502. ' };',
  12503. ' this.DoVirtual = function (pA,pB) {',
  12504. ' };',
  12505. ' this.DoIt = function (pA,pB) {',
  12506. ' };',
  12507. ' this.DoIt2 = function (pA,pB) {',
  12508. ' };',
  12509. ' this.GetIt = function (pA, pB) {',
  12510. ' var Result = 0;',
  12511. ' return Result;',
  12512. ' };',
  12513. '});',
  12514. 'rtl.createClass(this, "TClassA", this.TObject, function () {',
  12515. ' this.DoAbstract = function (pA,pB) {',
  12516. ' $mod.TObject.DoVirtual.call(this,pA,pB);',
  12517. ' $mod.TObject.DoVirtual.call(this,pA,0);',
  12518. ' };',
  12519. ' this.DoVirtual = function (pA,pB) {',
  12520. ' $mod.TObject.DoVirtual.apply(this, arguments);',
  12521. ' $mod.TObject.DoVirtual.call(this,pA,pB);',
  12522. ' $mod.TObject.DoVirtual.call(this,pA,0);',
  12523. ' this.DoIt(pA,pB);',
  12524. ' this.DoIt(pA,0);',
  12525. ' this.DoIt2(pA,2);',
  12526. ' this.DoIt2(1,2);',
  12527. ' };',
  12528. ' this.GetIt$1 = function (pA, pB) {',
  12529. ' var Result = 0;',
  12530. ' pA = $mod.TObject.GetIt.apply(this, arguments);',
  12531. ' return Result;',
  12532. ' };',
  12533. '});'
  12534. ]),
  12535. LinesToStr([ // this.$main
  12536. ''
  12537. ]));
  12538. end;
  12539. procedure TTestModule.TestClasS_CallInheritedConstructor;
  12540. begin
  12541. StartProgram(false);
  12542. Add('type');
  12543. Add(' TObject = class');
  12544. Add(' constructor Create; virtual;');
  12545. Add(' constructor CreateWithB(b: boolean);');
  12546. Add(' end;');
  12547. Add(' TA = class');
  12548. Add(' constructor Create; override;');
  12549. Add(' constructor CreateWithC(c: char);');
  12550. Add(' procedure DoIt;');
  12551. Add(' class function DoSome: TObject;');
  12552. Add(' end;');
  12553. Add('constructor tobject.create;');
  12554. Add('begin');
  12555. Add(' inherited; // call non existing ancestor -> ignore silently');
  12556. Add('end;');
  12557. Add('constructor tobject.createwithb(b: boolean);');
  12558. Add('begin');
  12559. Add(' inherited; // call non existing ancestor -> ignore silently');
  12560. Add(' create; // normal call');
  12561. Add('end;');
  12562. Add('constructor ta.create;');
  12563. Add('begin');
  12564. Add(' inherited; // normal call TObject.Create');
  12565. Add(' inherited create; // normal call TObject.Create');
  12566. Add(' inherited createwithb(false); // normal call TObject.CreateWithB');
  12567. Add('end;');
  12568. Add('constructor ta.createwithc(c: char);');
  12569. Add('begin');
  12570. Add(' inherited create; // call TObject.Create');
  12571. Add(' inherited createwithb(true); // call TObject.CreateWithB');
  12572. Add(' doit;');
  12573. Add(' doit();');
  12574. Add(' dosome;');
  12575. Add('end;');
  12576. Add('procedure ta.doit;');
  12577. Add('begin');
  12578. Add(' create; // normal call');
  12579. Add(' createwithb(false); // normal call');
  12580. Add(' createwithc(''c''); // normal call');
  12581. Add('end;');
  12582. Add('class function ta.dosome: TObject;');
  12583. Add('begin');
  12584. Add(' Result:=create; // constructor');
  12585. Add(' Result:=createwithb(true); // constructor');
  12586. Add(' Result:=createwithc(''c''); // constructor');
  12587. Add('end;');
  12588. Add('begin');
  12589. ConvertProgram;
  12590. CheckSource('TestClass_CallInheritedConstructor',
  12591. LinesToStr([ // statements
  12592. 'rtl.createClass(this,"TObject",null,function(){',
  12593. ' this.$init = function () {',
  12594. ' };',
  12595. ' this.$final = function () {',
  12596. ' };',
  12597. ' this.Create = function () {',
  12598. ' return this;',
  12599. ' };',
  12600. ' this.CreateWithB = function (b) {',
  12601. ' this.Create();',
  12602. ' return this;',
  12603. ' };',
  12604. '});',
  12605. 'rtl.createClass(this, "TA", this.TObject, function () {',
  12606. ' this.Create = function () {',
  12607. ' $mod.TObject.Create.call(this);',
  12608. ' $mod.TObject.Create.call(this);',
  12609. ' $mod.TObject.CreateWithB.call(this, false);',
  12610. ' return this;',
  12611. ' };',
  12612. ' this.CreateWithC = function (c) {',
  12613. ' $mod.TObject.Create.call(this);',
  12614. ' $mod.TObject.CreateWithB.call(this, true);',
  12615. ' this.DoIt();',
  12616. ' this.DoIt();',
  12617. ' this.$class.DoSome();',
  12618. ' return this;',
  12619. ' };',
  12620. ' this.DoIt = function () {',
  12621. ' this.Create();',
  12622. ' this.CreateWithB(false);',
  12623. ' this.CreateWithC("c");',
  12624. ' };',
  12625. ' this.DoSome = function () {',
  12626. ' var Result = null;',
  12627. ' Result = this.$create("Create");',
  12628. ' Result = this.$create("CreateWithB", [true]);',
  12629. ' Result = this.$create("CreateWithC", ["c"]);',
  12630. ' return Result;',
  12631. ' };',
  12632. '});'
  12633. ]),
  12634. LinesToStr([ // this.$main
  12635. ''
  12636. ]));
  12637. end;
  12638. procedure TTestModule.TestClass_ClassVar_Assign;
  12639. begin
  12640. StartProgram(false);
  12641. Add([
  12642. 'type',
  12643. ' TObject = class',
  12644. ' public',
  12645. ' class var vI: longint;',
  12646. ' class var Sub: TObject;',
  12647. ' constructor Create;',
  12648. ' class function GetIt(var Par: longint): tobject;',
  12649. ' end;',
  12650. 'constructor tobject.create;',
  12651. 'begin',
  12652. ' vi:=vi+1;',
  12653. ' Self.vi:=Self.vi+1;',
  12654. ' inc(vi);',
  12655. 'end;',
  12656. 'class function tobject.getit(var par: longint): tobject;',
  12657. 'begin',
  12658. ' vi:=vi+3;',
  12659. ' Self.vi:=Self.vi+4;',
  12660. ' inc(vi);',
  12661. ' Result:=self.sub;',
  12662. ' GetIt(vi);',
  12663. 'end;',
  12664. 'var Obj: tobject;',
  12665. 'begin',
  12666. ' obj:=tobject.create;',
  12667. ' tobject.vi:=3;',
  12668. ' if tobject.vi=4 then ;',
  12669. ' tobject.sub:=nil;',
  12670. ' obj.sub:=nil;',
  12671. ' obj.sub.sub:=nil;']);
  12672. ConvertProgram;
  12673. CheckSource('TestClass_ClassVar_Assign',
  12674. LinesToStr([ // statements
  12675. 'rtl.createClass(this,"TObject",null,function(){',
  12676. ' this.vI = 0;',
  12677. ' this.Sub = null;',
  12678. ' this.$init = function () {',
  12679. ' };',
  12680. ' this.$final = function () {',
  12681. ' };',
  12682. ' this.Create = function(){',
  12683. ' $mod.TObject.vI = this.vI+1;',
  12684. ' $mod.TObject.vI = this.vI+1;',
  12685. ' $mod.TObject.vI += 1;',
  12686. ' return this;',
  12687. ' };',
  12688. ' this.GetIt = function(Par){',
  12689. ' var Result = null;',
  12690. ' $mod.TObject.vI = this.vI + 3;',
  12691. ' $mod.TObject.vI = this.vI + 4;',
  12692. ' $mod.TObject.vI += 1;',
  12693. ' Result = this.Sub;',
  12694. ' this.GetIt({',
  12695. ' p: $mod.TObject,',
  12696. ' get: function () {',
  12697. ' return this.p.vI;',
  12698. ' },',
  12699. ' set: function (v) {',
  12700. ' this.p.vI = v;',
  12701. ' }',
  12702. ' });',
  12703. ' return Result;',
  12704. ' };',
  12705. '});',
  12706. 'this.Obj = null;'
  12707. ]),
  12708. LinesToStr([ // $mod.$main
  12709. '$mod.Obj = $mod.TObject.$create("Create");',
  12710. '$mod.TObject.vI = 3;',
  12711. 'if ($mod.TObject.vI === 4);',
  12712. '$mod.TObject.Sub=null;',
  12713. '$mod.TObject.Sub=null;',
  12714. '$mod.TObject.Sub=null;',
  12715. '']));
  12716. end;
  12717. procedure TTestModule.TestClass_CallClassMethod;
  12718. begin
  12719. StartProgram(false);
  12720. Add('type');
  12721. Add(' TObject = class');
  12722. Add(' public');
  12723. Add(' class var vI: longint;');
  12724. Add(' class var Sub: TObject;');
  12725. Add(' constructor Create;');
  12726. Add(' function GetMore(Par: longint): longint;');
  12727. Add(' class function GetIt(Par: longint): tobject;');
  12728. Add(' end;');
  12729. Add('constructor tobject.create;');
  12730. Add('begin');
  12731. Add(' sub:=getit(3);');
  12732. Add(' vi:=getmore(4);');
  12733. Add(' sub:=Self.getit(5);');
  12734. Add(' vi:=Self.getmore(6);');
  12735. Add('end;');
  12736. Add('function tobject.getmore(par: longint): longint;');
  12737. Add('begin');
  12738. Add(' sub:=getit(11);');
  12739. Add(' vi:=getmore(12);');
  12740. Add(' sub:=self.getit(13);');
  12741. Add(' vi:=self.getmore(14);');
  12742. Add('end;');
  12743. Add('class function tobject.getit(par: longint): tobject;');
  12744. Add('begin');
  12745. Add(' sub:=getit(21);');
  12746. Add(' vi:=sub.getmore(22);');
  12747. Add(' sub:=self.getit(23);');
  12748. Add(' vi:=self.sub.getmore(24);');
  12749. Add('end;');
  12750. Add('var Obj: tobject;');
  12751. Add('begin');
  12752. Add(' obj:=tobject.create;');
  12753. Add(' tobject.getit(5);');
  12754. Add(' obj.getit(6);');
  12755. Add(' obj.sub.getit(7);');
  12756. Add(' obj.sub.getit(8).SUB:=nil;');
  12757. Add(' obj.sub.getit(9).GETIT(10);');
  12758. Add(' obj.sub.getit(11).SuB.getit(12);');
  12759. ConvertProgram;
  12760. CheckSource('TestClass_CallClassMethod',
  12761. LinesToStr([ // statements
  12762. 'rtl.createClass(this,"TObject",null,function(){',
  12763. ' this.vI = 0;',
  12764. ' this.Sub = null;',
  12765. ' this.$init = function () {',
  12766. ' };',
  12767. ' this.$final = function () {',
  12768. ' };',
  12769. ' this.Create = function(){',
  12770. ' $mod.TObject.Sub = this.$class.GetIt(3);',
  12771. ' $mod.TObject.vI = this.GetMore(4);',
  12772. ' $mod.TObject.Sub = this.$class.GetIt(5);',
  12773. ' $mod.TObject.vI = this.GetMore(6);',
  12774. ' return this;',
  12775. ' };',
  12776. ' this.GetMore = function(Par){',
  12777. ' var Result = 0;',
  12778. ' $mod.TObject.Sub = this.$class.GetIt(11);',
  12779. ' $mod.TObject.vI = this.GetMore(12);',
  12780. ' $mod.TObject.Sub = this.$class.GetIt(13);',
  12781. ' $mod.TObject.vI = this.GetMore(14);',
  12782. ' return Result;',
  12783. ' };',
  12784. ' this.GetIt = function(Par){',
  12785. ' var Result = null;',
  12786. ' $mod.TObject.Sub = this.GetIt(21);',
  12787. ' $mod.TObject.vI = this.Sub.GetMore(22);',
  12788. ' $mod.TObject.Sub = this.GetIt(23);',
  12789. ' $mod.TObject.vI = this.Sub.GetMore(24);',
  12790. ' return Result;',
  12791. ' };',
  12792. '});',
  12793. 'this.Obj = null;'
  12794. ]),
  12795. LinesToStr([ // $mod.$main
  12796. '$mod.Obj = $mod.TObject.$create("Create");',
  12797. '$mod.TObject.GetIt(5);',
  12798. '$mod.Obj.$class.GetIt(6);',
  12799. '$mod.Obj.Sub.$class.GetIt(7);',
  12800. '$mod.TObject.Sub=null;',
  12801. '$mod.Obj.Sub.$class.GetIt(9).$class.GetIt(10);',
  12802. '$mod.Obj.Sub.$class.GetIt(11).Sub.$class.GetIt(12);',
  12803. '']));
  12804. end;
  12805. procedure TTestModule.TestClass_Property;
  12806. begin
  12807. StartProgram(false);
  12808. Add('type');
  12809. Add(' TObject = class');
  12810. Add(' Fx: longint;');
  12811. Add(' Fy: longint;');
  12812. Add(' function GetInt: longint;');
  12813. Add(' procedure SetInt(Value: longint);');
  12814. Add(' procedure DoIt;');
  12815. Add(' property IntA: longint read Fx write Fy;');
  12816. Add(' property IntB: longint read GetInt write SetInt;');
  12817. Add(' end;');
  12818. Add('function tobject.getint: longint;');
  12819. Add('begin');
  12820. Add(' result:=fx;');
  12821. Add('end;');
  12822. Add('procedure tobject.setint(value: longint);');
  12823. Add('begin');
  12824. Add(' if value=fy then exit;');
  12825. Add(' fy:=value;');
  12826. Add('end;');
  12827. Add('procedure tobject.doit;');
  12828. Add('begin');
  12829. Add(' IntA:=IntA+1;');
  12830. Add(' Self.IntA:=Self.IntA+1;');
  12831. Add(' IntB:=IntB+1;');
  12832. Add(' Self.IntB:=Self.IntB+1;');
  12833. Add('end;');
  12834. Add('var Obj: tobject;');
  12835. Add('begin');
  12836. Add(' obj.inta:=obj.inta+1;');
  12837. Add(' if obj.intb=2 then;');
  12838. Add(' obj.intb:=obj.intb+2;');
  12839. Add(' obj.setint(obj.inta);');
  12840. ConvertProgram;
  12841. CheckSource('TestClass_Property',
  12842. LinesToStr([ // statements
  12843. 'rtl.createClass(this, "TObject", null, function () {',
  12844. ' this.$init = function () {',
  12845. ' this.Fx = 0;',
  12846. ' this.Fy = 0;',
  12847. ' };',
  12848. ' this.$final = function () {',
  12849. ' };',
  12850. ' this.GetInt = function () {',
  12851. ' var Result = 0;',
  12852. ' Result = this.Fx;',
  12853. ' return Result;',
  12854. ' };',
  12855. ' this.SetInt = function (Value) {',
  12856. ' if (Value === this.Fy) return;',
  12857. ' this.Fy = Value;',
  12858. ' };',
  12859. ' this.DoIt = function () {',
  12860. ' this.Fy = this.Fx + 1;',
  12861. ' this.Fy = this.Fx + 1;',
  12862. ' this.SetInt(this.GetInt() + 1);',
  12863. ' this.SetInt(this.GetInt() + 1);',
  12864. ' };',
  12865. '});',
  12866. 'this.Obj = null;'
  12867. ]),
  12868. LinesToStr([ // $mod.$main
  12869. '$mod.Obj.Fy = $mod.Obj.Fx + 1;',
  12870. 'if ($mod.Obj.GetInt() === 2);',
  12871. '$mod.Obj.SetInt($mod.Obj.GetInt() + 2);',
  12872. '$mod.Obj.SetInt($mod.Obj.Fx);'
  12873. ]));
  12874. end;
  12875. procedure TTestModule.TestClass_Property_ClassMethod;
  12876. begin
  12877. StartProgram(false);
  12878. Add([
  12879. 'type',
  12880. ' TObject = class',
  12881. ' class var Fx: longint;',
  12882. ' class var Fy: longint;',
  12883. ' class function GetInt: longint;',
  12884. ' class procedure SetInt(Value: longint);',
  12885. ' end;',
  12886. ' TBird = class',
  12887. ' class procedure DoIt;',
  12888. ' class property IntA: longint read Fx write Fy;',
  12889. ' class property IntB: longint read GetInt write SetInt;',
  12890. ' end;',
  12891. 'class function tobject.getint: longint;',
  12892. 'begin',
  12893. ' result:=fx;',
  12894. 'end;',
  12895. 'class procedure tobject.setint(value: longint);',
  12896. 'begin',
  12897. 'end;',
  12898. 'class procedure tbird.doit;',
  12899. 'begin',
  12900. ' FX:=3;',
  12901. ' IntA:=IntA+1;',
  12902. ' Self.IntA:=Self.IntA+1;',
  12903. ' IntB:=IntB+1;',
  12904. ' Self.IntB:=Self.IntB+1;',
  12905. ' with Self do begin',
  12906. ' FX:=11;',
  12907. ' IntA:=IntA+12;',
  12908. ' IntB:=IntB+13;',
  12909. ' end;',
  12910. 'end;',
  12911. 'var Obj: tbird;',
  12912. 'begin',
  12913. ' tbird.fx:=tbird.fx+1;',
  12914. ' tbird.inta:=tbird.inta+1;',
  12915. ' if tbird.intb=2 then;',
  12916. ' tbird.intb:=tbird.intb+2;',
  12917. ' tbird.setint(tbird.inta);',
  12918. ' obj.inta:=obj.inta+1;',
  12919. ' if obj.intb=2 then;',
  12920. ' obj.intb:=obj.intb+2;',
  12921. ' obj.setint(obj.inta);',
  12922. ' with Tbird do begin',
  12923. ' FX:=FY+1;',
  12924. ' inta:=inta+2;',
  12925. ' intb:=intb+3;',
  12926. ' end;',
  12927. ' with Obj do begin',
  12928. ' FX:=FY+1;',
  12929. ' inta:=inta+2;',
  12930. ' intb:=intb+3;',
  12931. ' end;',
  12932. '']);
  12933. ConvertProgram;
  12934. CheckSource('TestClass_Property_ClassMethod',
  12935. LinesToStr([ // statements
  12936. 'rtl.createClass(this, "TObject", null, function () {',
  12937. ' this.Fx = 0;',
  12938. ' this.Fy = 0;',
  12939. ' this.$init = function () {',
  12940. ' };',
  12941. ' this.$final = function () {',
  12942. ' };',
  12943. ' this.GetInt = function () {',
  12944. ' var Result = 0;',
  12945. ' Result = this.Fx;',
  12946. ' return Result;',
  12947. ' };',
  12948. ' this.SetInt = function (Value) {',
  12949. ' };',
  12950. '});',
  12951. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  12952. ' this.DoIt = function () {',
  12953. ' $mod.TObject.Fx = 3;',
  12954. ' $mod.TObject.Fy = this.Fx + 1;',
  12955. ' $mod.TObject.Fy = this.Fx + 1;',
  12956. ' this.SetInt(this.GetInt() + 1);',
  12957. ' this.SetInt(this.GetInt() + 1);',
  12958. ' $mod.TObject.Fx = 11;',
  12959. ' $mod.TObject.Fy = this.Fx + 12;',
  12960. ' this.SetInt(this.GetInt() + 13);',
  12961. ' };',
  12962. '});',
  12963. 'this.Obj = null;'
  12964. ]),
  12965. LinesToStr([ // $mod.$main
  12966. '$mod.TObject.Fx = $mod.TBird.Fx + 1;',
  12967. '$mod.TObject.Fy = $mod.TBird.Fx + 1;',
  12968. 'if ($mod.TBird.GetInt() === 2);',
  12969. '$mod.TBird.SetInt($mod.TBird.GetInt() + 2);',
  12970. '$mod.TBird.SetInt($mod.TBird.Fx);',
  12971. '$mod.TObject.Fy = $mod.Obj.Fx + 1;',
  12972. 'if ($mod.Obj.$class.GetInt() === 2);',
  12973. '$mod.Obj.$class.SetInt($mod.Obj.$class.GetInt() + 2);',
  12974. '$mod.Obj.$class.SetInt($mod.Obj.Fx);',
  12975. 'var $with = $mod.TBird;',
  12976. '$mod.TObject.Fx = $with.Fy + 1;',
  12977. '$mod.TObject.Fy = $with.Fx + 2;',
  12978. '$with.SetInt($with.GetInt() + 3);',
  12979. 'var $with1 = $mod.Obj;',
  12980. '$mod.TObject.Fx = $with1.Fy + 1;',
  12981. '$mod.TObject.Fy = $with1.Fx + 2;',
  12982. '$with1.$class.SetInt($with1.$class.GetInt() + 3);',
  12983. '']));
  12984. end;
  12985. procedure TTestModule.TestClass_Property_Indexed;
  12986. begin
  12987. StartProgram(false);
  12988. Add([
  12989. 'type',
  12990. ' TObject = class',
  12991. ' FItems: array of longint;',
  12992. ' function GetItems(Index: longint): longint;',
  12993. ' procedure SetItems(Index: longint; Value: longint);',
  12994. ' procedure DoIt;',
  12995. ' property Items[Index: longint]: longint read getitems write setitems;',
  12996. ' end;',
  12997. 'function tobject.getitems(index: longint): longint;',
  12998. 'begin',
  12999. ' Result:=fitems[index];',
  13000. 'end;',
  13001. 'procedure tobject.setitems(index: longint; value: longint);',
  13002. 'begin',
  13003. ' fitems[index]:=value;',
  13004. 'end;',
  13005. 'procedure tobject.doit;',
  13006. 'begin',
  13007. ' items[1]:=2;',
  13008. ' items[3]:=items[4];',
  13009. ' self.items[5]:=self.items[6];',
  13010. ' items[items[7]]:=items[items[8]];',
  13011. 'end;',
  13012. 'var Obj: tobject;',
  13013. 'begin',
  13014. ' obj.Items[11]:=obj.Items[12];',
  13015. '']);
  13016. ConvertProgram;
  13017. CheckSource('TestClass_Property_Indexed',
  13018. LinesToStr([ // statements
  13019. 'rtl.createClass(this, "TObject", null, function () {',
  13020. ' this.$init = function () {',
  13021. ' this.FItems = [];',
  13022. ' };',
  13023. ' this.$final = function () {',
  13024. ' this.FItems = undefined;',
  13025. ' };',
  13026. ' this.GetItems = function (Index) {',
  13027. ' var Result = 0;',
  13028. ' Result = this.FItems[Index];',
  13029. ' return Result;',
  13030. ' };',
  13031. ' this.SetItems = function (Index, Value) {',
  13032. ' this.FItems[Index] = Value;',
  13033. ' };',
  13034. ' this.DoIt = function () {',
  13035. ' this.SetItems(1, 2);',
  13036. ' this.SetItems(3,this.GetItems(4));',
  13037. ' this.SetItems(5,this.GetItems(6));',
  13038. ' this.SetItems(this.GetItems(7), this.GetItems(this.GetItems(8)));',
  13039. ' };',
  13040. '});',
  13041. 'this.Obj = null;'
  13042. ]),
  13043. LinesToStr([ // $mod.$main
  13044. '$mod.Obj.SetItems(11,$mod.Obj.GetItems(12));'
  13045. ]));
  13046. end;
  13047. procedure TTestModule.TestClass_Property_IndexSpec;
  13048. begin
  13049. StartProgram(false);
  13050. Add([
  13051. 'type',
  13052. ' TEnum = (red, blue);',
  13053. ' TObject = class',
  13054. ' function GetIntBool(Index: longint): boolean; virtual; abstract;',
  13055. ' procedure SetIntBool(Index: longint; b: boolean); virtual; abstract;',
  13056. ' function GetEnumBool(Index: TEnum): boolean; virtual; abstract;',
  13057. ' procedure SetEnumBool(Index: TEnum; b: boolean); virtual; abstract;',
  13058. ' function GetStrIntBool(A: String; I: longint): boolean; virtual; abstract;',
  13059. ' procedure SetStrIntBool(A: String; I: longint; b: boolean); virtual; abstract;',
  13060. ' property B1: boolean index 1 read GetIntBool write SetIntBool;',
  13061. ' property B2: boolean index TEnum.blue read GetEnumBool write SetEnumBool;',
  13062. ' property B3: boolean index ord(red) read GetIntBool write SetIntBool;',
  13063. ' property I1[A: String]: boolean index ord(blue) read GetStrIntBool write SetStrIntBool;',
  13064. ' end;',
  13065. 'procedure DoIt(b: boolean); begin end;',
  13066. 'var',
  13067. ' o: TObject;',
  13068. 'begin',
  13069. ' o.B1:=o.B1;',
  13070. ' o.B2:=o.B2;',
  13071. ' o.B3:=o.B3;',
  13072. ' o.I1[''a'']:=o.I1[''b''];',
  13073. ' doit(o.b1);',
  13074. ' doit(o.b2);',
  13075. ' doit(o.i1[''c'']);',
  13076. '']);
  13077. ConvertProgram;
  13078. CheckSource('TestClass_Property_IndexSpec',
  13079. LinesToStr([ // statements
  13080. 'this.TEnum = {',
  13081. ' "0": "red",',
  13082. ' red: 0,',
  13083. ' "1": "blue",',
  13084. ' blue: 1',
  13085. '};',
  13086. 'rtl.createClass(this, "TObject", null, function () {',
  13087. ' this.$init = function () {',
  13088. ' };',
  13089. ' this.$final = function () {',
  13090. ' };',
  13091. '});',
  13092. 'this.DoIt = function (b) {',
  13093. '};',
  13094. 'this.o = null;',
  13095. '']),
  13096. LinesToStr([ // $mod.$main
  13097. '$mod.o.SetIntBool(1, $mod.o.GetIntBool(1));',
  13098. '$mod.o.SetEnumBool($mod.TEnum.blue, $mod.o.GetEnumBool($mod.TEnum.blue));',
  13099. '$mod.o.SetIntBool(0, $mod.o.GetIntBool(0));',
  13100. '$mod.o.SetStrIntBool("a", 1, $mod.o.GetStrIntBool("b", 1));',
  13101. '$mod.DoIt($mod.o.GetIntBool(1));',
  13102. '$mod.DoIt($mod.o.GetEnumBool($mod.TEnum.blue));',
  13103. '$mod.DoIt($mod.o.GetStrIntBool("c", 1));',
  13104. '']));
  13105. end;
  13106. procedure TTestModule.TestClass_PropertyOfTypeArray;
  13107. begin
  13108. StartProgram(false);
  13109. Add('type');
  13110. Add(' TArray = array of longint;');
  13111. Add(' TObject = class');
  13112. Add(' FItems: TArray;');
  13113. Add(' function GetItems: tarray;');
  13114. Add(' procedure SetItems(Value: tarray);');
  13115. Add(' property Items: tarray read getitems write setitems;');
  13116. Add(' procedure SetNumbers(const Value: tarray);');
  13117. Add(' property Numbers: tarray write setnumbers;');
  13118. Add(' end;');
  13119. Add('function tobject.getitems: tarray;');
  13120. Add('begin');
  13121. Add(' Result:=fitems;');
  13122. Add('end;');
  13123. Add('procedure tobject.setitems(value: tarray);');
  13124. Add('begin');
  13125. Add(' fitems:=value;');
  13126. Add(' fitems:=nil;');
  13127. Add(' Items:=nil;');
  13128. Add(' Items:=Items;');
  13129. Add(' Items[1]:=2;');
  13130. Add(' fitems[3]:=Items[4];');
  13131. Add(' Items[5]:=Items[6];');
  13132. Add(' Self.Items[7]:=8;');
  13133. Add(' Self.Items[9]:=Self.Items[10];');
  13134. Add(' Items[Items[11]]:=Items[Items[12]];');
  13135. Add('end;');
  13136. Add('procedure tobject.SetNumbers(const Value: tarray);');
  13137. Add('begin;');
  13138. Add(' Numbers:=nil;');
  13139. Add(' Numbers:=Value;');
  13140. Add(' Self.Numbers:=Value;');
  13141. Add('end;');
  13142. Add('var Obj: tobject;');
  13143. Add('begin');
  13144. Add(' obj.items:=nil;');
  13145. Add(' obj.items:=obj.items;');
  13146. Add(' obj.items[11]:=obj.items[12];');
  13147. ConvertProgram;
  13148. CheckSource('TestClass_PropertyOfTypeArray',
  13149. LinesToStr([ // statements
  13150. 'rtl.createClass(this, "TObject", null, function () {',
  13151. ' this.$init = function () {',
  13152. ' this.FItems = [];',
  13153. ' };',
  13154. ' this.$final = function () {',
  13155. ' this.FItems = undefined;',
  13156. ' };',
  13157. ' this.GetItems = function () {',
  13158. ' var Result = [];',
  13159. ' Result = rtl.arrayRef(this.FItems);',
  13160. ' return Result;',
  13161. ' };',
  13162. ' this.SetItems = function (Value) {',
  13163. ' this.FItems = rtl.arrayRef(Value);',
  13164. ' this.FItems = [];',
  13165. ' this.SetItems([]);',
  13166. ' this.SetItems(rtl.arrayRef(this.GetItems()));',
  13167. ' this.GetItems()[1] = 2;',
  13168. ' this.FItems[3] = this.GetItems()[4];',
  13169. ' this.GetItems()[5] = this.GetItems()[6];',
  13170. ' this.GetItems()[7] = 8;',
  13171. ' this.GetItems()[9] = this.GetItems()[10];',
  13172. ' this.GetItems()[this.GetItems()[11]] = this.GetItems()[this.GetItems()[12]];',
  13173. ' };',
  13174. ' this.SetNumbers = function (Value) {',
  13175. ' this.SetNumbers([]);',
  13176. ' this.SetNumbers(Value);',
  13177. ' this.SetNumbers(Value);',
  13178. ' };',
  13179. '});',
  13180. 'this.Obj = null;'
  13181. ]),
  13182. LinesToStr([ // $mod.$main
  13183. '$mod.Obj.SetItems([]);',
  13184. '$mod.Obj.SetItems($mod.Obj.GetItems());',
  13185. '$mod.Obj.GetItems()[11] = $mod.Obj.GetItems()[12];'
  13186. ]));
  13187. end;
  13188. procedure TTestModule.TestClass_PropertyDefault;
  13189. begin
  13190. StartProgram(false);
  13191. Add([
  13192. 'type',
  13193. ' TArray = array of longint;',
  13194. ' TObject = class',
  13195. ' end;',
  13196. ' TBird = class',
  13197. ' FItems: TArray;',
  13198. ' function GetItems(Index: longint): longint;',
  13199. ' procedure SetItems(Index, Value: longint);',
  13200. ' property Items[Index: longint]: longint read getitems write setitems; default;',
  13201. ' end;',
  13202. 'function TBird.getitems(index: longint): longint;',
  13203. 'begin',
  13204. 'end;',
  13205. 'procedure TBird.setitems(index, value: longint);',
  13206. 'begin',
  13207. ' Self[1]:=2;',
  13208. ' Self[3]:=Self[index];',
  13209. ' Self[index]:=Self[Self[value]];',
  13210. ' Self[Self[4]]:=value;',
  13211. 'end;',
  13212. 'var',
  13213. ' Bird: TBird;',
  13214. ' Obj: TObject;',
  13215. 'begin',
  13216. ' bird[11]:=12;',
  13217. ' bird[13]:=bird[14];',
  13218. ' bird[Bird[15]]:=bird[Bird[15]];',
  13219. ' TBird(obj)[16]:=TBird(obj)[17];',
  13220. ' (obj as tbird)[18]:=19;',
  13221. '']);
  13222. ConvertProgram;
  13223. CheckSource('TestClass_PropertyDefault',
  13224. LinesToStr([ // statements
  13225. 'rtl.createClass(this, "TObject", null, function () {',
  13226. ' this.$init = function () {',
  13227. ' };',
  13228. ' this.$final = function () {',
  13229. ' };',
  13230. '});',
  13231. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  13232. ' this.$init = function () {',
  13233. ' $mod.TObject.$init.call(this);',
  13234. ' this.FItems = [];',
  13235. ' };',
  13236. ' this.$final = function () {',
  13237. ' this.FItems = undefined;',
  13238. ' $mod.TObject.$final.call(this);',
  13239. ' };',
  13240. ' this.GetItems = function (Index) {',
  13241. ' var Result = 0;',
  13242. ' return Result;',
  13243. ' };',
  13244. ' this.SetItems = function (Index, Value) {',
  13245. ' this.SetItems(1, 2);',
  13246. ' this.SetItems(3, this.GetItems(Index));',
  13247. ' this.SetItems(Index, this.GetItems(this.GetItems(Value)));',
  13248. ' this.SetItems(this.GetItems(4), Value);',
  13249. ' };',
  13250. '});',
  13251. 'this.Bird = null;',
  13252. 'this.Obj = null;',
  13253. '']),
  13254. LinesToStr([ // $mod.$main
  13255. '$mod.Bird.SetItems(11, 12);',
  13256. '$mod.Bird.SetItems(13, $mod.Bird.GetItems(14));',
  13257. '$mod.Bird.SetItems($mod.Bird.GetItems(15), $mod.Bird.GetItems($mod.Bird.GetItems(15)));',
  13258. '$mod.Obj.SetItems(16, $mod.Obj.GetItems(17));',
  13259. 'rtl.as($mod.Obj, $mod.TBird).SetItems(18, 19);',
  13260. '']));
  13261. end;
  13262. procedure TTestModule.TestClass_PropertyDefault_TypecastToOtherDefault;
  13263. begin
  13264. StartProgram(false);
  13265. Add([
  13266. 'type',
  13267. ' TObject = class end;',
  13268. ' TAlphaList = class',
  13269. ' function GetAlphas(Index: boolean): Pointer; virtual; abstract;',
  13270. ' procedure SetAlphas(Index: boolean; Value: Pointer); virtual; abstract;',
  13271. ' property Alphas[Index: boolean]: Pointer read getAlphas write setAlphas; default;',
  13272. ' end;',
  13273. ' TBetaList = class',
  13274. ' function GetBetas(Index: longint): Pointer; virtual; abstract;',
  13275. ' procedure SetBetas(Index: longint; Value: Pointer); virtual; abstract;',
  13276. ' property Betas[Index: longint]: Pointer read getBetas write setBetas; default;',
  13277. ' end;',
  13278. ' TBird = class',
  13279. ' procedure DoIt;',
  13280. ' end;',
  13281. 'procedure TBird.DoIt;',
  13282. 'var',
  13283. ' List: TAlphaList;',
  13284. 'begin',
  13285. ' if TBetaList(List[true])[3]=nil then ;',
  13286. ' TBetaList(List[false])[5]:=nil;',
  13287. 'end;',
  13288. 'var',
  13289. ' List: TAlphaList;',
  13290. 'begin',
  13291. ' if TBetaList(List[true])[3]=nil then ;',
  13292. ' TBetaList(List[false])[5]:=nil;',
  13293. '']);
  13294. ConvertProgram;
  13295. CheckSource('TestClass_PropertyDefault_TypecastToOtherDefault',
  13296. LinesToStr([ // statements
  13297. 'rtl.createClass(this, "TObject", null, function () {',
  13298. ' this.$init = function () {',
  13299. ' };',
  13300. ' this.$final = function () {',
  13301. ' };',
  13302. '});',
  13303. 'rtl.createClass(this, "TAlphaList", this.TObject, function () {',
  13304. '});',
  13305. 'rtl.createClass(this, "TBetaList", this.TObject, function () {',
  13306. '});',
  13307. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  13308. ' this.DoIt = function () {',
  13309. ' var List = null;',
  13310. ' if (List.GetAlphas(true).GetBetas(3) === null) ;',
  13311. ' List.GetAlphas(false).SetBetas(5, null);',
  13312. ' };',
  13313. '});',
  13314. 'this.List = null;',
  13315. '']),
  13316. LinesToStr([ // $mod.$main
  13317. 'if ($mod.List.GetAlphas(true).GetBetas(3) === null) ;',
  13318. '$mod.List.GetAlphas(false).SetBetas(5, null);',
  13319. '']));
  13320. end;
  13321. procedure TTestModule.TestClass_PropertyOverride;
  13322. begin
  13323. StartProgram(false);
  13324. Add('type');
  13325. Add(' integer = longint;');
  13326. Add(' TObject = class');
  13327. Add(' FItem: integer;');
  13328. Add(' function GetItem: integer; external name ''GetItem'';');
  13329. Add(' procedure SetItem(Value: integer); external name ''SetItem'';');
  13330. Add(' property Item: integer read getitem write setitem;');
  13331. Add(' end;');
  13332. Add(' TCar = class');
  13333. Add(' FBag: integer;');
  13334. Add(' function GetBag: integer; external name ''GetBag'';');
  13335. Add(' property Item read getbag;');
  13336. Add(' end;');
  13337. Add('var');
  13338. Add(' Obj: tobject;');
  13339. Add(' Car: tcar;');
  13340. Add('begin');
  13341. Add(' Obj.Item:=Obj.Item;');
  13342. Add(' Car.Item:=Car.Item;');
  13343. ConvertProgram;
  13344. CheckSource('TestClass_PropertyOverride',
  13345. LinesToStr([ // statements
  13346. 'rtl.createClass(this, "TObject", null, function () {',
  13347. ' this.$init = function () {',
  13348. ' this.FItem = 0;',
  13349. ' };',
  13350. ' this.$final = function () {',
  13351. ' };',
  13352. '});',
  13353. 'rtl.createClass(this, "TCar", this.TObject, function () {',
  13354. ' this.$init = function () {',
  13355. ' $mod.TObject.$init.call(this);',
  13356. ' this.FBag = 0;',
  13357. ' };',
  13358. '});',
  13359. 'this.Obj = null;',
  13360. 'this.Car = null;',
  13361. '']),
  13362. LinesToStr([ // $mod.$main
  13363. '$mod.Obj.SetItem($mod.Obj.GetItem());',
  13364. '$mod.Car.SetItem($mod.Car.GetBag());',
  13365. '']));
  13366. end;
  13367. procedure TTestModule.TestClass_PropertyIncVisibility;
  13368. begin
  13369. AddModuleWithIntfImplSrc('unit1.pp',
  13370. LinesToStr([
  13371. 'type',
  13372. ' TNumber = longint;',
  13373. ' TInteger = longint;',
  13374. ' TObject = class',
  13375. ' private',
  13376. ' function GetItems(Index: TNumber): TInteger; virtual; abstract;',
  13377. ' procedure SetItems(Index: TInteger; Value: TNumber); virtual; abstract;',
  13378. ' protected',
  13379. ' property Items[Index: TNumber]: longint read GetItems write SetItems;',
  13380. ' end;']),
  13381. LinesToStr([
  13382. '']));
  13383. StartProgram(true);
  13384. Add([
  13385. 'uses unit1;',
  13386. 'type',
  13387. ' TBird = class',
  13388. ' public',
  13389. ' property Items;',
  13390. ' end;',
  13391. 'procedure DoIt(i: TInteger);',
  13392. 'begin',
  13393. 'end;',
  13394. 'var b: TBird;',
  13395. 'begin',
  13396. ' b.Items[1]:=2;',
  13397. ' b.Items[3]:=b.Items[4];',
  13398. ' DoIt(b.Items[5]);',
  13399. '']);
  13400. ConvertProgram;
  13401. CheckSource('TestClass_PropertyIncVisibility',
  13402. LinesToStr([ // statements
  13403. 'rtl.createClass(this, "TBird", pas.unit1.TObject, function () {',
  13404. '});',
  13405. 'this.DoIt = function (i) {',
  13406. '};',
  13407. 'this.b = null;'
  13408. ]),
  13409. LinesToStr([ // $mod.$main
  13410. '$mod.b.SetItems(1, 2);',
  13411. '$mod.b.SetItems(3, $mod.b.GetItems(4));',
  13412. '$mod.DoIt($mod.b.GetItems(5));'
  13413. ]));
  13414. end;
  13415. procedure TTestModule.TestClass_Assigned;
  13416. begin
  13417. StartProgram(false);
  13418. Add('type');
  13419. Add(' TObject = class');
  13420. Add(' end;');
  13421. Add('var');
  13422. Add(' Obj: tobject;');
  13423. Add(' b: boolean;');
  13424. Add('begin');
  13425. Add(' if Assigned(obj) then ;');
  13426. Add(' b:=Assigned(obj) or false;');
  13427. ConvertProgram;
  13428. CheckSource('TestClass_Assigned',
  13429. LinesToStr([ // statements
  13430. 'rtl.createClass(this, "TObject", null, function () {',
  13431. ' this.$init = function () {',
  13432. ' };',
  13433. ' this.$final = function () {',
  13434. ' };',
  13435. '});',
  13436. 'this.Obj = null;',
  13437. 'this.b = false;'
  13438. ]),
  13439. LinesToStr([ // $mod.$main
  13440. 'if ($mod.Obj != null);',
  13441. '$mod.b = ($mod.Obj != null) || false;'
  13442. ]));
  13443. end;
  13444. procedure TTestModule.TestClass_WithClassDoCreate;
  13445. begin
  13446. StartProgram(false);
  13447. Add('type');
  13448. Add(' TObject = class');
  13449. Add(' aBool: boolean;');
  13450. Add(' Arr: array of boolean;');
  13451. Add(' constructor Create;');
  13452. Add(' end;');
  13453. Add('constructor TObject.Create; begin end;');
  13454. Add('var');
  13455. Add(' Obj: tobject;');
  13456. Add(' b: boolean;');
  13457. Add('begin');
  13458. Add(' with tobject.create do begin');
  13459. Add(' b:=abool;');
  13460. Add(' abool:=b;');
  13461. Add(' b:=arr[1];');
  13462. Add(' arr[2]:=b;');
  13463. Add(' end;');
  13464. Add(' with tobject do');
  13465. Add(' obj:=create;');
  13466. Add(' with obj do begin');
  13467. Add(' create;');
  13468. Add(' b:=abool;');
  13469. Add(' abool:=b;');
  13470. Add(' b:=arr[3];');
  13471. Add(' arr[4]:=b;');
  13472. Add(' end;');
  13473. ConvertProgram;
  13474. CheckSource('TestClass_WithClassDoCreate',
  13475. LinesToStr([ // statements
  13476. 'rtl.createClass(this, "TObject", null, function () {',
  13477. ' this.$init = function () {',
  13478. ' this.aBool = false;',
  13479. ' this.Arr = [];',
  13480. ' };',
  13481. ' this.$final = function () {',
  13482. ' this.Arr = undefined;',
  13483. ' };',
  13484. ' this.Create = function () {',
  13485. ' return this;',
  13486. ' };',
  13487. '});',
  13488. 'this.Obj = null;',
  13489. 'this.b = false;'
  13490. ]),
  13491. LinesToStr([ // $mod.$main
  13492. 'var $with = $mod.TObject.$create("Create");',
  13493. '$mod.b = $with.aBool;',
  13494. '$with.aBool = $mod.b;',
  13495. '$mod.b = $with.Arr[1];',
  13496. '$with.Arr[2] = $mod.b;',
  13497. 'var $with1 = $mod.TObject;',
  13498. '$mod.Obj = $with1.$create("Create");',
  13499. 'var $with2 = $mod.Obj;',
  13500. '$with2.Create();',
  13501. '$mod.b = $with2.aBool;',
  13502. '$with2.aBool = $mod.b;',
  13503. '$mod.b = $with2.Arr[3];',
  13504. '$with2.Arr[4] = $mod.b;',
  13505. '']));
  13506. end;
  13507. procedure TTestModule.TestClass_WithClassInstDoProperty;
  13508. begin
  13509. StartProgram(false);
  13510. Add('type');
  13511. Add(' TObject = class');
  13512. Add(' FInt: longint;');
  13513. Add(' constructor Create;');
  13514. Add(' function GetSize: longint;');
  13515. Add(' procedure SetSize(Value: longint);');
  13516. Add(' property Int: longint read FInt write FInt;');
  13517. Add(' property Size: longint read GetSize write SetSize;');
  13518. Add(' end;');
  13519. Add('constructor TObject.Create; begin end;');
  13520. Add('function TObject.GetSize: longint; begin; end;');
  13521. Add('procedure TObject.SetSize(Value: longint); begin; end;');
  13522. Add('var');
  13523. Add(' Obj: tobject;');
  13524. Add(' i: longint;');
  13525. Add('begin');
  13526. Add(' with TObject.Create do begin');
  13527. Add(' i:=int;');
  13528. Add(' int:=i;');
  13529. Add(' i:=size;');
  13530. Add(' size:=i;');
  13531. Add(' end;');
  13532. Add(' with obj do begin');
  13533. Add(' i:=int;');
  13534. Add(' int:=i;');
  13535. Add(' i:=size;');
  13536. Add(' size:=i;');
  13537. Add(' end;');
  13538. ConvertProgram;
  13539. CheckSource('TestClass_WithClassInstDoProperty',
  13540. LinesToStr([ // statements
  13541. 'rtl.createClass(this, "TObject", null, function () {',
  13542. ' this.$init = function () {',
  13543. ' this.FInt = 0;',
  13544. ' };',
  13545. ' this.$final = function () {',
  13546. ' };',
  13547. ' this.Create = function () {',
  13548. ' return this;',
  13549. ' };',
  13550. ' this.GetSize = function () {',
  13551. ' var Result = 0;',
  13552. ' return Result;',
  13553. ' };',
  13554. ' this.SetSize = function (Value) {',
  13555. ' };',
  13556. '});',
  13557. 'this.Obj = null;',
  13558. 'this.i = 0;'
  13559. ]),
  13560. LinesToStr([ // $mod.$main
  13561. 'var $with = $mod.TObject.$create("Create");',
  13562. '$mod.i = $with.FInt;',
  13563. '$with.FInt = $mod.i;',
  13564. '$mod.i = $with.GetSize();',
  13565. '$with.SetSize($mod.i);',
  13566. 'var $with1 = $mod.Obj;',
  13567. '$mod.i = $with1.FInt;',
  13568. '$with1.FInt = $mod.i;',
  13569. '$mod.i = $with1.GetSize();',
  13570. '$with1.SetSize($mod.i);',
  13571. '']));
  13572. end;
  13573. procedure TTestModule.TestClass_WithClassInstDoPropertyWithParams;
  13574. begin
  13575. StartProgram(false);
  13576. Add('type');
  13577. Add(' TObject = class');
  13578. Add(' constructor Create;');
  13579. Add(' function GetItems(Index: longint): longint;');
  13580. Add(' procedure SetItems(Index, Value: longint);');
  13581. Add(' property Items[Index: longint]: longint read GetItems write SetItems;');
  13582. Add(' end;');
  13583. Add('constructor TObject.Create; begin end;');
  13584. Add('function tobject.getitems(index: longint): longint; begin; end;');
  13585. Add('procedure tobject.setitems(index, value: longint); begin; end;');
  13586. Add('var');
  13587. Add(' Obj: tobject;');
  13588. Add(' i: longint;');
  13589. Add('begin');
  13590. Add(' with TObject.Create do begin');
  13591. Add(' i:=Items[1];');
  13592. Add(' Items[2]:=i;');
  13593. Add(' end;');
  13594. Add(' with obj do begin');
  13595. Add(' i:=Items[3];');
  13596. Add(' Items[4]:=i;');
  13597. Add(' end;');
  13598. ConvertProgram;
  13599. CheckSource('TestClass_WithClassInstDoPropertyWithParams',
  13600. LinesToStr([ // statements
  13601. 'rtl.createClass(this, "TObject", null, function () {',
  13602. ' this.$init = function () {',
  13603. ' };',
  13604. ' this.$final = function () {',
  13605. ' };',
  13606. ' this.Create = function () {',
  13607. ' return this;',
  13608. ' };',
  13609. ' this.GetItems = function (Index) {',
  13610. ' var Result = 0;',
  13611. ' return Result;',
  13612. ' };',
  13613. ' this.SetItems = function (Index, Value) {',
  13614. ' };',
  13615. '});',
  13616. 'this.Obj = null;',
  13617. 'this.i = 0;'
  13618. ]),
  13619. LinesToStr([ // $mod.$main
  13620. 'var $with = $mod.TObject.$create("Create");',
  13621. '$mod.i = $with.GetItems(1);',
  13622. '$with.SetItems(2, $mod.i);',
  13623. 'var $with1 = $mod.Obj;',
  13624. '$mod.i = $with1.GetItems(3);',
  13625. '$with1.SetItems(4, $mod.i);',
  13626. '']));
  13627. end;
  13628. procedure TTestModule.TestClass_WithClassInstDoFunc;
  13629. begin
  13630. StartProgram(false);
  13631. Add('type');
  13632. Add(' TObject = class');
  13633. Add(' constructor Create;');
  13634. Add(' function GetSize: longint;');
  13635. Add(' procedure SetSize(Value: longint);');
  13636. Add(' end;');
  13637. Add('constructor TObject.Create; begin end;');
  13638. Add('function TObject.GetSize: longint; begin; end;');
  13639. Add('procedure TObject.SetSize(Value: longint); begin; end;');
  13640. Add('var');
  13641. Add(' Obj: tobject;');
  13642. Add(' i: longint;');
  13643. Add('begin');
  13644. Add(' with TObject.Create do begin');
  13645. Add(' i:=GetSize;');
  13646. Add(' i:=GetSize();');
  13647. Add(' SetSize(i);');
  13648. Add(' end;');
  13649. Add(' with obj do begin');
  13650. Add(' i:=GetSize;');
  13651. Add(' i:=GetSize();');
  13652. Add(' SetSize(i);');
  13653. Add(' end;');
  13654. ConvertProgram;
  13655. CheckSource('TestClass_WithClassInstDoFunc',
  13656. LinesToStr([ // statements
  13657. 'rtl.createClass(this, "TObject", null, function () {',
  13658. ' this.$init = function () {',
  13659. ' };',
  13660. ' this.$final = function () {',
  13661. ' };',
  13662. ' this.Create = function () {',
  13663. ' return this;',
  13664. ' };',
  13665. ' this.GetSize = function () {',
  13666. ' var Result = 0;',
  13667. ' return Result;',
  13668. ' };',
  13669. ' this.SetSize = function (Value) {',
  13670. ' };',
  13671. '});',
  13672. 'this.Obj = null;',
  13673. 'this.i = 0;'
  13674. ]),
  13675. LinesToStr([ // $mod.$main
  13676. 'var $with = $mod.TObject.$create("Create");',
  13677. '$mod.i = $with.GetSize();',
  13678. '$mod.i = $with.GetSize();',
  13679. '$with.SetSize($mod.i);',
  13680. 'var $with1 = $mod.Obj;',
  13681. '$mod.i = $with1.GetSize();',
  13682. '$mod.i = $with1.GetSize();',
  13683. '$with1.SetSize($mod.i);',
  13684. '']));
  13685. end;
  13686. procedure TTestModule.TestClass_TypeCast;
  13687. begin
  13688. StartProgram(false);
  13689. Add('type');
  13690. Add(' TObject = class');
  13691. Add(' Next: TObject;');
  13692. Add(' constructor Create;');
  13693. Add(' end;');
  13694. Add(' TControl = class(TObject)');
  13695. Add(' Arr: array of TObject;');
  13696. Add(' function GetIt(vI: longint = 0): TObject;');
  13697. Add(' end;');
  13698. Add('constructor tobject.create; begin end;');
  13699. Add('function tcontrol.getit(vi: longint = 0): tobject; begin end;');
  13700. Add('var');
  13701. Add(' Obj: tobject;');
  13702. Add('begin');
  13703. Add(' obj:=tcontrol(obj).next;');
  13704. Add(' tcontrol(obj):=nil;');
  13705. Add(' obj:=tcontrol(obj);');
  13706. Add(' tcontrol(obj):=tcontrol(tcontrol(obj).getit);');
  13707. Add(' tcontrol(obj):=tcontrol(tcontrol(obj).getit());');
  13708. Add(' tcontrol(obj):=tcontrol(tcontrol(obj).getit(1));');
  13709. Add(' tcontrol(obj):=tcontrol(tcontrol(tcontrol(obj).getit).arr[2]);');
  13710. Add(' obj:=tcontrol(nil);');
  13711. ConvertProgram;
  13712. CheckSource('TestClass_TypeCast',
  13713. LinesToStr([ // statements
  13714. 'rtl.createClass(this, "TObject", null, function () {',
  13715. ' this.$init = function () {',
  13716. ' this.Next = null;',
  13717. ' };',
  13718. ' this.$final = function () {',
  13719. ' this.Next = undefined;',
  13720. ' };',
  13721. ' this.Create = function () {',
  13722. ' return this;',
  13723. ' };',
  13724. '});',
  13725. 'rtl.createClass(this, "TControl", this.TObject, function () {',
  13726. ' this.$init = function () {',
  13727. ' $mod.TObject.$init.call(this);',
  13728. ' this.Arr = [];',
  13729. ' };',
  13730. ' this.$final = function () {',
  13731. ' this.Arr = undefined;',
  13732. ' $mod.TObject.$final.call(this);',
  13733. ' };',
  13734. ' this.GetIt = function (vI) {',
  13735. ' var Result = null;',
  13736. ' return Result;',
  13737. ' };',
  13738. '});',
  13739. 'this.Obj = null;'
  13740. ]),
  13741. LinesToStr([ // $mod.$main
  13742. '$mod.Obj = $mod.Obj.Next;',
  13743. '$mod.Obj = null;',
  13744. '$mod.Obj = $mod.Obj;',
  13745. '$mod.Obj = $mod.Obj.GetIt(0);',
  13746. '$mod.Obj = $mod.Obj.GetIt(0);',
  13747. '$mod.Obj = $mod.Obj.GetIt(1);',
  13748. '$mod.Obj = $mod.Obj.GetIt(0).Arr[2];',
  13749. '$mod.Obj = null;',
  13750. '']));
  13751. end;
  13752. procedure TTestModule.TestClass_TypeCastUntypedParam;
  13753. begin
  13754. StartProgram(false);
  13755. Add('type');
  13756. Add(' TObject = class end;');
  13757. Add('procedure ProcA(var A);');
  13758. Add('begin');
  13759. Add(' TObject(A):=nil;');
  13760. Add(' TObject(A):=TObject(A);');
  13761. Add(' if TObject(A)=nil then ;');
  13762. Add(' if nil=TObject(A) then ;');
  13763. Add('end;');
  13764. Add('procedure ProcB(out A);');
  13765. Add('begin');
  13766. Add(' TObject(A):=nil;');
  13767. Add(' TObject(A):=TObject(A);');
  13768. Add(' if TObject(A)=nil then ;');
  13769. Add(' if nil=TObject(A) then ;');
  13770. Add('end;');
  13771. Add('procedure ProcC(const A);');
  13772. Add('begin');
  13773. Add(' if TObject(A)=nil then ;');
  13774. Add(' if nil=TObject(A) then ;');
  13775. Add('end;');
  13776. Add('var o: TObject;');
  13777. Add('begin');
  13778. Add(' ProcA(o);');
  13779. Add(' ProcB(o);');
  13780. Add(' ProcC(o);');
  13781. ConvertProgram;
  13782. CheckSource('TestClass_TypeCastUntypedParam',
  13783. LinesToStr([ // statements
  13784. 'rtl.createClass(this, "TObject", null, function () {',
  13785. ' this.$init = function () {',
  13786. ' };',
  13787. ' this.$final = function () {',
  13788. ' };',
  13789. '});',
  13790. 'this.ProcA = function (A) {',
  13791. ' A.set(null);',
  13792. ' A.set(A.get());',
  13793. ' if (A.get() === null);',
  13794. ' if (null === A.get());',
  13795. '};',
  13796. 'this.ProcB = function (A) {',
  13797. ' A.set(null);',
  13798. ' A.set(A.get());',
  13799. ' if (A.get() === null);',
  13800. ' if (null === A.get());',
  13801. '};',
  13802. 'this.ProcC = function (A) {',
  13803. ' if (A === null);',
  13804. ' if (null === A);',
  13805. '};',
  13806. 'this.o = null;',
  13807. '']),
  13808. LinesToStr([ // $mod.$main
  13809. '$mod.ProcA({',
  13810. ' p: $mod,',
  13811. ' get: function () {',
  13812. ' return this.p.o;',
  13813. ' },',
  13814. ' set: function (v) {',
  13815. ' this.p.o = v;',
  13816. ' }',
  13817. '});',
  13818. '$mod.ProcB({',
  13819. ' p: $mod,',
  13820. ' get: function () {',
  13821. ' return this.p.o;',
  13822. ' },',
  13823. ' set: function (v) {',
  13824. ' this.p.o = v;',
  13825. ' }',
  13826. '});',
  13827. '$mod.ProcC($mod.o);',
  13828. '']));
  13829. end;
  13830. procedure TTestModule.TestClass_Overloads;
  13831. begin
  13832. StartProgram(false);
  13833. Add('type');
  13834. Add(' TObject = class');
  13835. Add(' procedure DoIt;');
  13836. Add(' procedure DoIt(vI: longint);');
  13837. Add(' end;');
  13838. Add('procedure TObject.DoIt;');
  13839. Add('begin');
  13840. Add(' DoIt;');
  13841. Add(' DoIt(1);');
  13842. Add('end;');
  13843. Add('procedure TObject.DoIt(vI: longint); begin end;');
  13844. Add('begin');
  13845. ConvertProgram;
  13846. CheckSource('TestClass_Overloads',
  13847. LinesToStr([ // statements
  13848. 'rtl.createClass(this, "TObject", null, function () {',
  13849. ' this.$init = function () {',
  13850. ' };',
  13851. ' this.$final = function () {',
  13852. ' };',
  13853. ' this.DoIt = function () {',
  13854. ' this.DoIt();',
  13855. ' this.DoIt$1(1);',
  13856. ' };',
  13857. ' this.DoIt$1 = function (vI) {',
  13858. ' };',
  13859. '});',
  13860. '']),
  13861. LinesToStr([ // $mod.$main
  13862. '']));
  13863. end;
  13864. procedure TTestModule.TestClass_OverloadsAncestor;
  13865. begin
  13866. StartProgram(false);
  13867. Add('type');
  13868. Add(' TObject = class;');
  13869. Add(' TObject = class');
  13870. Add(' procedure DoIt(vA: longint);');
  13871. Add(' procedure DoIt(vA, vB: longint);');
  13872. Add(' end;');
  13873. Add(' TCar = class;');
  13874. Add(' TCar = class');
  13875. Add(' procedure DoIt(vA: longint);');
  13876. Add(' procedure DoIt(vA, vB: longint);');
  13877. Add(' end;');
  13878. Add('procedure tobject.doit(va: longint);');
  13879. Add('begin');
  13880. Add(' doit(1);');
  13881. Add(' doit(1,2);');
  13882. Add('end;');
  13883. Add('procedure tobject.doit(va, vb: longint); begin end;');
  13884. Add('procedure tcar.doit(va: longint);');
  13885. Add('begin');
  13886. Add(' doit(1);');
  13887. Add(' doit(1,2);');
  13888. Add(' inherited doit(1);');
  13889. Add(' inherited doit(1,2);');
  13890. Add('end;');
  13891. Add('procedure tcar.doit(va, vb: longint); begin end;');
  13892. Add('begin');
  13893. ConvertProgram;
  13894. CheckSource('TestClass_OverloadsAncestor',
  13895. LinesToStr([ // statements
  13896. 'rtl.createClass(this, "TObject", null, function () {',
  13897. ' this.$init = function () {',
  13898. ' };',
  13899. ' this.$final = function () {',
  13900. ' };',
  13901. ' this.DoIt = function (vA) {',
  13902. ' this.DoIt(1);',
  13903. ' this.DoIt$1(1,2);',
  13904. ' };',
  13905. ' this.DoIt$1 = function (vA, vB) {',
  13906. ' };',
  13907. '});',
  13908. 'rtl.createClass(this, "TCar", this.TObject, function () {',
  13909. ' this.DoIt$2 = function (vA) {',
  13910. ' this.DoIt$2(1);',
  13911. ' this.DoIt$3(1, 2);',
  13912. ' $mod.TObject.DoIt.call(this, 1);',
  13913. ' $mod.TObject.DoIt$1.call(this, 1, 2);',
  13914. ' };',
  13915. ' this.DoIt$3 = function (vA, vB) {',
  13916. ' };',
  13917. '});',
  13918. '']),
  13919. LinesToStr([ // $mod.$main
  13920. '']));
  13921. end;
  13922. procedure TTestModule.TestClass_OverloadConstructor;
  13923. begin
  13924. StartProgram(false);
  13925. Add('type');
  13926. Add(' TObject = class');
  13927. Add(' constructor Create(vA: longint);');
  13928. Add(' constructor Create(vA, vB: longint);');
  13929. Add(' end;');
  13930. Add(' TCar = class');
  13931. Add(' constructor Create(vA: longint);');
  13932. Add(' constructor Create(vA, vB: longint);');
  13933. Add(' end;');
  13934. Add('constructor tobject.create(va: longint);');
  13935. Add('begin');
  13936. Add(' create(1);');
  13937. Add(' create(1,2);');
  13938. Add('end;');
  13939. Add('constructor tobject.create(va, vb: longint); begin end;');
  13940. Add('constructor tcar.create(va: longint);');
  13941. Add('begin');
  13942. Add(' create(1);');
  13943. Add(' create(1,2);');
  13944. Add(' inherited create(1);');
  13945. Add(' inherited create(1,2);');
  13946. Add('end;');
  13947. Add('constructor tcar.create(va, vb: longint); begin end;');
  13948. Add('begin');
  13949. Add(' tobject.create(1);');
  13950. Add(' tobject.create(1,2);');
  13951. Add(' tcar.create(1);');
  13952. Add(' tcar.create(1,2);');
  13953. ConvertProgram;
  13954. CheckSource('TestClass_OverloadConstructor',
  13955. LinesToStr([ // statements
  13956. 'rtl.createClass(this, "TObject", null, function () {',
  13957. ' this.$init = function () {',
  13958. ' };',
  13959. ' this.$final = function () {',
  13960. ' };',
  13961. ' this.Create = function (vA) {',
  13962. ' this.Create(1);',
  13963. ' this.Create$1(1,2);',
  13964. ' return this;',
  13965. ' };',
  13966. ' this.Create$1 = function (vA, vB) {',
  13967. ' return this;',
  13968. ' };',
  13969. '});',
  13970. 'rtl.createClass(this, "TCar", this.TObject, function () {',
  13971. ' this.Create$2 = function (vA) {',
  13972. ' this.Create$2(1);',
  13973. ' this.Create$3(1, 2);',
  13974. ' $mod.TObject.Create.call(this, 1);',
  13975. ' $mod.TObject.Create$1.call(this, 1, 2);',
  13976. ' return this;',
  13977. ' };',
  13978. ' this.Create$3 = function (vA, vB) {',
  13979. ' return this;',
  13980. ' };',
  13981. '});',
  13982. '']),
  13983. LinesToStr([ // $mod.$main
  13984. '$mod.TObject.$create("Create", [1]);',
  13985. '$mod.TObject.$create("Create$1", [1, 2]);',
  13986. '$mod.TCar.$create("Create$2", [1]);',
  13987. '$mod.TCar.$create("Create$3", [1, 2]);',
  13988. '']));
  13989. end;
  13990. procedure TTestModule.TestClass_OverloadDelphiOverride;
  13991. begin
  13992. StartProgram(false);
  13993. Add([
  13994. '{$mode delphi}',
  13995. 'type',
  13996. ' TObject = class end;',
  13997. ' TBird = class',
  13998. ' function {#a}GetValue: longint; overload; virtual;',
  13999. ' function {#b}GetValue(AValue: longint): longint; overload; virtual;',
  14000. ' end;',
  14001. ' TEagle = class(TBird)',
  14002. ' function {#c}GetValue: longint; overload; override;',
  14003. ' function {#d}GetValue(AValue: longint): longint; overload; override;',
  14004. ' end;',
  14005. 'function TBird.GetValue: longint;',
  14006. 'begin',
  14007. ' if 3={@a}GetValue then ;',
  14008. ' if 4={@b}GetValue(5) then ;',
  14009. 'end;',
  14010. 'function TBird.GetValue(AValue: longint): longint;',
  14011. 'begin',
  14012. 'end;',
  14013. 'function TEagle.GetValue: longint;',
  14014. 'begin',
  14015. ' if 13={@c}GetValue then ;',
  14016. ' if 14={@d}GetValue(15) then ;',
  14017. ' if 15=inherited {@a}GetValue then ;',
  14018. ' if 16=inherited {@b}GetValue(17) then ;',
  14019. 'end;',
  14020. 'function TEagle.GetValue(AValue: longint): longint;',
  14021. 'begin',
  14022. 'end;',
  14023. 'var',
  14024. ' e: TEagle;',
  14025. 'begin',
  14026. ' if 23=e.{@c}GetValue then ;',
  14027. ' if 24=e.{@d}GetValue(25) then ;']);
  14028. ConvertProgram;
  14029. CheckSource('TestClass_OverloadDelphiOverride',
  14030. LinesToStr([ // statements
  14031. 'rtl.createClass(this, "TObject", null, function () {',
  14032. ' this.$init = function () {',
  14033. ' };',
  14034. ' this.$final = function () {',
  14035. ' };',
  14036. '});',
  14037. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  14038. ' this.GetValue = function () {',
  14039. ' var Result = 0;',
  14040. ' if (3 === this.GetValue()) ;',
  14041. ' if (4 === this.GetValue$1(5)) ;',
  14042. ' return Result;',
  14043. ' };',
  14044. ' this.GetValue$1 = function (AValue) {',
  14045. ' var Result = 0;',
  14046. ' return Result;',
  14047. ' };',
  14048. '});',
  14049. 'rtl.createClass(this, "TEagle", this.TBird, function () {',
  14050. ' this.GetValue = function () {',
  14051. ' var Result = 0;',
  14052. ' if (13 === this.GetValue()) ;',
  14053. ' if (14 === this.GetValue$1(15)) ;',
  14054. ' if (15 === $mod.TBird.GetValue.call(this)) ;',
  14055. ' if (16 === $mod.TBird.GetValue$1.call(this, 17)) ;',
  14056. ' return Result;',
  14057. ' };',
  14058. ' this.GetValue$1 = function (AValue) {',
  14059. ' var Result = 0;',
  14060. ' return Result;',
  14061. ' };',
  14062. '});',
  14063. 'this.e = null;',
  14064. '']),
  14065. LinesToStr([ // $mod.$main
  14066. 'if (23 === $mod.e.GetValue()) ;',
  14067. 'if (24 === $mod.e.GetValue$1(25)) ;',
  14068. '']));
  14069. end;
  14070. procedure TTestModule.TestClass_ReintroduceVarDelphi;
  14071. begin
  14072. StartProgram(false);
  14073. Add([
  14074. '{$mode delphi}',
  14075. 'type',
  14076. ' TObject = class end;',
  14077. ' TAnimal = class',
  14078. ' public',
  14079. ' {#animal_a}A: longint;',
  14080. ' function {#animal_b}B: longint;',
  14081. ' end;',
  14082. ' TBird = class(TAnimal)',
  14083. ' public',
  14084. ' {#bird_a}A: double;',
  14085. ' {#bird_b}B: boolean;',
  14086. ' end;',
  14087. ' TEagle = class(TBird)',
  14088. ' public',
  14089. ' function {#eagle_a}A: boolean;',
  14090. ' {#eagle_b}B: double;',
  14091. ' end;',
  14092. 'function TAnimal.B: longint;',
  14093. 'begin',
  14094. 'end;',
  14095. 'function TEagle.A: boolean;',
  14096. 'begin',
  14097. ' {@eagle_b}B:=3.3;',
  14098. ' {@eagle_a}A();',
  14099. ' TBird(Self).{@bird_b}B:=true;',
  14100. ' TAnimal(Self).{@animal_a}A:=17;',
  14101. ' inherited {@bird_b}B:=inherited {bird_a}A>1;', // Delphi allows only inherited <functionname>
  14102. 'end;',
  14103. 'var',
  14104. ' e: TEagle;',
  14105. 'begin',
  14106. ' e.{@eagle_b}B:=5.3;',
  14107. ' if e.{@eagle_a}A then ;',
  14108. '']);
  14109. ConvertProgram;
  14110. CheckSource('TestClass_ReintroduceVarDelphi',
  14111. LinesToStr([ // statements
  14112. 'rtl.createClass(this, "TObject", null, function () {',
  14113. ' this.$init = function () {',
  14114. ' };',
  14115. ' this.$final = function () {',
  14116. ' };',
  14117. '});',
  14118. 'rtl.createClass(this, "TAnimal", this.TObject, function () {',
  14119. ' this.$init = function () {',
  14120. ' $mod.TObject.$init.call(this);',
  14121. ' this.A = 0;',
  14122. ' };',
  14123. ' this.B = function () {',
  14124. ' var Result = 0;',
  14125. ' return Result;',
  14126. ' };',
  14127. '});',
  14128. 'rtl.createClass(this, "TBird", this.TAnimal, function () {',
  14129. ' this.$init = function () {',
  14130. ' $mod.TAnimal.$init.call(this);',
  14131. ' this.A$1 = 0.0;',
  14132. ' this.B$1 = false;',
  14133. ' };',
  14134. '});',
  14135. 'rtl.createClass(this, "TEagle", this.TBird, function () {',
  14136. ' this.$init = function () {',
  14137. ' $mod.TBird.$init.call(this);',
  14138. ' this.B$2 = 0.0;',
  14139. ' };',
  14140. ' this.A$2 = function () {',
  14141. ' var Result = false;',
  14142. ' this.B$2 = 3.3;',
  14143. ' this.A$2();',
  14144. ' this.B$1 = true;',
  14145. ' this.A = 17;',
  14146. ' this.B$1 = this.A$1 > 1;',
  14147. ' return Result;',
  14148. ' };',
  14149. '});',
  14150. 'this.e = null;',
  14151. '']),
  14152. LinesToStr([ // $mod.$main
  14153. '$mod.e.B$2 = 5.3;',
  14154. 'if ($mod.e.A$2()) ;',
  14155. '']));
  14156. end;
  14157. procedure TTestModule.TestClass_ReintroducedVar;
  14158. begin
  14159. StartProgram(false);
  14160. Add('type');
  14161. Add(' TObject = class');
  14162. Add(' strict private');
  14163. Add(' Some: longint;');
  14164. Add(' end;');
  14165. Add(' TMobile = class');
  14166. Add(' strict private');
  14167. Add(' Some: string;');
  14168. Add(' end;');
  14169. Add(' TCar = class(tmobile)');
  14170. Add(' procedure Some;');
  14171. Add(' procedure Some(vA: longint);');
  14172. Add(' end;');
  14173. Add('procedure tcar.some;');
  14174. Add('begin');
  14175. Add(' Some;');
  14176. Add(' Some(1);');
  14177. Add('end;');
  14178. Add('procedure tcar.some(va: longint); begin end;');
  14179. Add('begin');
  14180. ConvertProgram;
  14181. CheckSource('TestClass_ReintroducedVar',
  14182. LinesToStr([ // statements
  14183. 'rtl.createClass(this, "TObject", null, function () {',
  14184. ' this.$init = function () {',
  14185. ' this.Some = 0;',
  14186. ' };',
  14187. ' this.$final = function () {',
  14188. ' };',
  14189. '});',
  14190. 'rtl.createClass(this, "TMobile", this.TObject, function () {',
  14191. ' this.$init = function () {',
  14192. ' $mod.TObject.$init.call(this);',
  14193. ' this.Some$1 = "";',
  14194. ' };',
  14195. '});',
  14196. 'rtl.createClass(this, "TCar", this.TMobile, function () {',
  14197. ' this.Some$2 = function () {',
  14198. ' this.Some$2();',
  14199. ' this.Some$3(1);',
  14200. ' };',
  14201. ' this.Some$3 = function (vA) {',
  14202. ' };',
  14203. '});',
  14204. '']),
  14205. LinesToStr([ // $mod.$main
  14206. '']));
  14207. end;
  14208. procedure TTestModule.TestClass_RaiseDescendant;
  14209. begin
  14210. StartProgram(false);
  14211. Add([
  14212. 'type',
  14213. ' TObject = class',
  14214. ' constructor Create(Msg: string);',
  14215. ' end;',
  14216. ' Exception = class',
  14217. ' end;',
  14218. ' EConvertError = class(Exception)',
  14219. ' end;',
  14220. 'constructor TObject.Create(Msg: string); begin end;',
  14221. 'function AssertConv(Msg: string = ''def''): EConvertError; begin end;',
  14222. 'begin',
  14223. ' raise Exception.Create(''Bar1'');',
  14224. ' raise EConvertError.Create(''Bar2'');',
  14225. ' raise AssertConv(''Bar2'');',
  14226. ' raise AssertConv;',
  14227. '']);
  14228. ConvertProgram;
  14229. CheckSource('TestClass_RaiseDescendant',
  14230. LinesToStr([ // statements
  14231. 'rtl.createClass(this, "TObject", null, function () {',
  14232. ' this.$init = function () {',
  14233. ' };',
  14234. ' this.$final = function () {',
  14235. ' };',
  14236. ' this.Create = function (Msg) {',
  14237. ' return this;',
  14238. ' };',
  14239. '});',
  14240. 'rtl.createClass(this, "Exception", this.TObject, function () {',
  14241. '});',
  14242. 'rtl.createClass(this, "EConvertError", this.Exception, function () {',
  14243. '});',
  14244. 'this.AssertConv = function (Msg) {',
  14245. ' var Result = null;',
  14246. ' return Result;',
  14247. '};',
  14248. '']),
  14249. LinesToStr([ // $mod.$main
  14250. 'throw $mod.Exception.$create("Create",["Bar1"]);',
  14251. 'throw $mod.EConvertError.$create("Create",["Bar2"]);',
  14252. 'throw $mod.AssertConv("Bar2");',
  14253. 'throw $mod.AssertConv("def");',
  14254. '']));
  14255. end;
  14256. procedure TTestModule.TestClass_ExternalMethod;
  14257. begin
  14258. AddModuleWithIntfImplSrc('unit2.pas',
  14259. LinesToStr([
  14260. 'type',
  14261. ' TObject = class',
  14262. ' public',
  14263. ' procedure Intern; external name ''$DoIntern'';',
  14264. ' end;',
  14265. '']),
  14266. LinesToStr([
  14267. '']));
  14268. StartUnit(true);
  14269. Add('interface');
  14270. Add('uses unit2;');
  14271. Add('type');
  14272. Add(' TCar = class(TObject)');
  14273. Add(' public');
  14274. Add(' procedure Intern2; external name ''$DoIntern2'';');
  14275. Add(' procedure DoIt;');
  14276. Add(' end;');
  14277. Add('implementation');
  14278. Add('procedure tcar.doit;');
  14279. Add('begin');
  14280. Add(' Intern;');
  14281. Add(' Intern();');
  14282. Add(' Intern2;');
  14283. Add(' Intern2();');
  14284. Add('end;');
  14285. Add('var Obj: TCar;');
  14286. Add('begin');
  14287. Add(' obj.intern;');
  14288. Add(' obj.intern();');
  14289. Add(' obj.intern2;');
  14290. Add(' obj.intern2();');
  14291. Add(' obj.doit;');
  14292. Add(' obj.doit();');
  14293. Add(' with obj do begin');
  14294. Add(' Intern;');
  14295. Add(' Intern();');
  14296. Add(' Intern2;');
  14297. Add(' Intern2();');
  14298. Add(' end;');
  14299. ConvertUnit;
  14300. CheckSource('TestClass_ExternalMethod',
  14301. LinesToStr([
  14302. 'var $impl = $mod.$impl;',
  14303. 'rtl.createClass(this, "TCar", pas.unit2.TObject, function () {',
  14304. ' this.DoIt = function () {',
  14305. ' this.$DoIntern();',
  14306. ' this.$DoIntern();',
  14307. ' this.$DoIntern2();',
  14308. ' this.$DoIntern2();',
  14309. ' };',
  14310. ' });',
  14311. '']),
  14312. LinesToStr([ // this.$init
  14313. '$impl.Obj.$DoIntern();',
  14314. '$impl.Obj.$DoIntern();',
  14315. '$impl.Obj.$DoIntern2();',
  14316. '$impl.Obj.$DoIntern2();',
  14317. '$impl.Obj.DoIt();',
  14318. '$impl.Obj.DoIt();',
  14319. 'var $with = $impl.Obj;',
  14320. '$with.$DoIntern();',
  14321. '$with.$DoIntern();',
  14322. '$with.$DoIntern2();',
  14323. '$with.$DoIntern2();',
  14324. '']),
  14325. LinesToStr([ // implementation
  14326. '$impl.Obj = null;',
  14327. '']) );
  14328. end;
  14329. procedure TTestModule.TestClass_ExternalVirtualNameMismatchFail;
  14330. begin
  14331. StartProgram(false);
  14332. Add('type');
  14333. Add(' TObject = class');
  14334. Add(' procedure DoIt; virtual; external name ''Foo'';');
  14335. Add(' end;');
  14336. Add('begin');
  14337. SetExpectedPasResolverError('Virtual method name must match external',
  14338. nVirtualMethodNameMustMatchExternal);
  14339. ConvertProgram;
  14340. end;
  14341. procedure TTestModule.TestClass_ExternalOverrideFail;
  14342. begin
  14343. StartProgram(false);
  14344. Add('type');
  14345. Add(' TObject = class');
  14346. Add(' procedure DoIt; virtual; external name ''DoIt'';');
  14347. Add(' end;');
  14348. Add(' TCar = class');
  14349. Add(' procedure DoIt; override; external name ''DoIt'';');
  14350. Add(' end;');
  14351. Add('begin');
  14352. SetExpectedPasResolverError('Invalid procedure modifier override,external',
  14353. nInvalidXModifierY);
  14354. ConvertProgram;
  14355. end;
  14356. procedure TTestModule.TestClass_ExternalVar;
  14357. begin
  14358. AddModuleWithIntfImplSrc('unit2.pas',
  14359. LinesToStr([
  14360. '{$modeswitch externalclass}',
  14361. 'type',
  14362. ' TObject = class',
  14363. ' public',
  14364. ' Intern: longint external name ''$Intern'';',
  14365. ' Bracket: longint external name ''["A B"]'';',
  14366. ' end;',
  14367. '']),
  14368. LinesToStr([
  14369. '']));
  14370. StartUnit(true);
  14371. Add([
  14372. 'interface',
  14373. 'uses unit2;',
  14374. '{$modeswitch externalclass}',
  14375. 'type',
  14376. ' TCar = class(tobject)',
  14377. ' public',
  14378. ' Intern2: longint external name ''$Intern2'';',
  14379. ' procedure DoIt;',
  14380. ' end;',
  14381. 'implementation',
  14382. 'procedure tcar.doit;',
  14383. 'begin',
  14384. ' Intern:=Intern+1;',
  14385. ' Intern2:=Intern2+2;',
  14386. ' Bracket:=Bracket+3;',
  14387. 'end;',
  14388. 'var Obj: TCar;',
  14389. 'begin',
  14390. ' obj.intern:=obj.intern+1;',
  14391. ' obj.intern2:=obj.intern2+2;',
  14392. ' obj.Bracket:=obj.Bracket+3;',
  14393. ' with obj do begin',
  14394. ' intern:=intern+1;',
  14395. ' intern2:=intern2+2;',
  14396. ' Bracket:=Bracket+3;',
  14397. ' end;']);
  14398. ConvertUnit;
  14399. CheckSource('TestClass_ExternalVar',
  14400. LinesToStr([
  14401. 'var $impl = $mod.$impl;',
  14402. 'rtl.createClass(this, "TCar", pas.unit2.TObject, function () {',
  14403. ' this.DoIt = function () {',
  14404. ' this.$Intern = this.$Intern + 1;',
  14405. ' this.$Intern2 = this.$Intern2 + 2;',
  14406. ' this["A B"] = this["A B"] + 3;',
  14407. ' };',
  14408. ' });',
  14409. '']),
  14410. LinesToStr([
  14411. '$impl.Obj.$Intern = $impl.Obj.$Intern + 1;',
  14412. '$impl.Obj.$Intern2 = $impl.Obj.$Intern2 + 2;',
  14413. '$impl.Obj["A B"] = $impl.Obj["A B"] + 3;',
  14414. 'var $with = $impl.Obj;',
  14415. '$with.$Intern = $with.$Intern + 1;',
  14416. '$with.$Intern2 = $with.$Intern2 + 2;',
  14417. '$with["A B"] = $with["A B"] + 3;',
  14418. '']),
  14419. LinesToStr([ // implementation
  14420. '$impl.Obj = null;',
  14421. '']));
  14422. end;
  14423. procedure TTestModule.TestClass_Const;
  14424. begin
  14425. StartProgram(false);
  14426. Add([
  14427. 'type',
  14428. ' integer = longint;',
  14429. ' TClass = class of TObject;',
  14430. ' TObject = class',
  14431. ' public',
  14432. ' const cI: integer = 3;',
  14433. ' procedure DoIt;',
  14434. ' class procedure DoMore;',
  14435. ' end;',
  14436. 'procedure tobject.doit;',
  14437. 'begin',
  14438. ' if cI=4 then;',
  14439. ' if 5=cI then;',
  14440. ' if Self.cI=6 then;',
  14441. ' if 7=Self.cI then;',
  14442. ' with Self do begin',
  14443. ' if cI=11 then;',
  14444. ' if 12=cI then;',
  14445. ' end;',
  14446. 'end;',
  14447. 'class procedure tobject.domore;',
  14448. 'begin',
  14449. ' if cI=8 then;',
  14450. ' if Self.cI=9 then;',
  14451. ' if 10=cI then;',
  14452. ' if 11=Self.cI then;',
  14453. ' with Self do begin',
  14454. ' if cI=13 then;',
  14455. ' if 14=cI then;',
  14456. ' end;',
  14457. 'end;',
  14458. 'var',
  14459. ' Obj: TObject;',
  14460. ' Cla: TClass;',
  14461. 'begin',
  14462. ' if TObject.cI=21 then ;',
  14463. ' if Obj.cI=22 then ;',
  14464. ' if Cla.cI=23 then ;',
  14465. ' with obj do if ci=24 then;',
  14466. ' with TObject do if ci=25 then;',
  14467. ' with Cla do if ci=26 then;']);
  14468. ConvertProgram;
  14469. CheckSource('TestClass_Const',
  14470. LinesToStr([
  14471. 'rtl.createClass(this, "TObject", null, function () {',
  14472. ' this.cI = 3;',
  14473. ' this.$init = function () {',
  14474. ' };',
  14475. ' this.$final = function () {',
  14476. ' };',
  14477. ' this.DoIt = function () {',
  14478. ' if (this.cI === 4) ;',
  14479. ' if (5 === this.cI) ;',
  14480. ' if (this.cI === 6) ;',
  14481. ' if (7 === this.cI) ;',
  14482. ' if (this.cI === 11) ;',
  14483. ' if (12 === this.cI) ;',
  14484. ' };',
  14485. ' this.DoMore = function () {',
  14486. ' if (this.cI === 8) ;',
  14487. ' if (this.cI === 9) ;',
  14488. ' if (10 === this.cI) ;',
  14489. ' if (11 === this.cI) ;',
  14490. ' if (this.cI === 13) ;',
  14491. ' if (14 === this.cI) ;',
  14492. ' };',
  14493. '});',
  14494. 'this.Obj = null;',
  14495. 'this.Cla = null;',
  14496. '']),
  14497. LinesToStr([
  14498. 'if ($mod.TObject.cI === 21) ;',
  14499. 'if ($mod.Obj.cI === 22) ;',
  14500. 'if ($mod.Cla.cI === 23) ;',
  14501. 'var $with = $mod.Obj;',
  14502. 'if ($with.cI === 24) ;',
  14503. 'var $with1 = $mod.TObject;',
  14504. 'if ($with1.cI === 25) ;',
  14505. 'var $with2 = $mod.Cla;',
  14506. 'if ($with2.cI === 26) ;',
  14507. '']));
  14508. end;
  14509. procedure TTestModule.TestClass_ConstEnum;
  14510. begin
  14511. StartProgram(false);
  14512. Add([
  14513. 'type',
  14514. ' TEnum = (red,blue);',
  14515. ' TObject = class',
  14516. ' end;',
  14517. ' TAnimal = class',
  14518. ' public',
  14519. ' type TSubEnum = (light,dark);',
  14520. ' const a = high(TEnum);',
  14521. ' const b = high(TSubEnum);',
  14522. ' end;',
  14523. ' TBird = class(TAnimal)',
  14524. ' public',
  14525. ' const c = high(TEnum);',
  14526. ' const d = high(TSubEnum);',
  14527. ' end;',
  14528. ' TAnt = class',
  14529. ' public',
  14530. ' const e = high(TEnum);',
  14531. ' const f = high(TBird.TSubEnum);',
  14532. ' end;',
  14533. 'begin',
  14534. '']);
  14535. ConvertProgram;
  14536. CheckSource('TestClass_ConstEnum',
  14537. LinesToStr([
  14538. 'this.TEnum = {',
  14539. ' "0": "red",',
  14540. ' red: 0,',
  14541. ' "1": "blue",',
  14542. ' blue: 1',
  14543. '};',
  14544. 'rtl.createClass(this, "TObject", null, function () {',
  14545. ' this.$init = function () {',
  14546. ' };',
  14547. ' this.$final = function () {',
  14548. ' };',
  14549. '});',
  14550. 'rtl.createClass(this, "TAnimal", this.TObject, function () {',
  14551. ' this.TSubEnum = {',
  14552. ' "0": "light",',
  14553. ' light: 0,',
  14554. ' "1": "dark",',
  14555. ' dark: 1',
  14556. ' };',
  14557. ' this.a = $mod.TEnum.blue;',
  14558. ' this.b = this.TSubEnum.dark;',
  14559. '});',
  14560. 'rtl.createClass(this, "TBird", this.TAnimal, function () {',
  14561. ' this.c = $mod.TEnum.blue;',
  14562. ' this.d = this.TSubEnum.dark;',
  14563. '});',
  14564. 'rtl.createClass(this, "TAnt", this.TObject, function () {',
  14565. ' this.e = $mod.TEnum.blue;',
  14566. ' this.f = $mod.TAnimal.TSubEnum.dark;',
  14567. '});',
  14568. '']),
  14569. LinesToStr([
  14570. '']));
  14571. end;
  14572. procedure TTestModule.TestClass_LocalConstDuplicate_Prg;
  14573. begin
  14574. StartProgram(false);
  14575. Add([
  14576. 'type',
  14577. ' TObject = class',
  14578. ' const cI: longint = 3;',
  14579. ' procedure Fly;',
  14580. ' procedure Run;',
  14581. ' end;',
  14582. ' TBird = class',
  14583. ' procedure Go;',
  14584. ' end;',
  14585. 'procedure tobject.fly;',
  14586. 'const cI: word = 4;',
  14587. 'begin',
  14588. ' if cI=Self.cI then ;',
  14589. 'end;',
  14590. 'procedure tobject.run;',
  14591. 'const cI: word = 5;',
  14592. 'begin',
  14593. ' if cI=Self.cI then ;',
  14594. 'end;',
  14595. 'procedure tbird.go;',
  14596. 'const cI: word = 6;',
  14597. 'begin',
  14598. ' if cI=Self.cI then ;',
  14599. 'end;',
  14600. 'begin',
  14601. '']);
  14602. ConvertProgram;
  14603. CheckSource('TestClass_LocalConstDuplicate_Prg',
  14604. LinesToStr([
  14605. 'rtl.createClass(this, "TObject", null, function () {',
  14606. ' this.cI = 3;',
  14607. ' this.$init = function () {',
  14608. ' };',
  14609. ' this.$final = function () {',
  14610. ' };',
  14611. ' var cI$1 = 4;',
  14612. ' this.Fly = function () {',
  14613. ' if (cI$1 === this.cI) ;',
  14614. ' };',
  14615. ' var cI$2 = 5;',
  14616. ' this.Run = function () {',
  14617. ' if (cI$2 === this.cI) ;',
  14618. ' };',
  14619. '});',
  14620. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  14621. ' var cI$3 = 6;',
  14622. ' this.Go = function () {',
  14623. ' if (cI$3 === this.cI) ;',
  14624. ' };',
  14625. '});',
  14626. '']),
  14627. LinesToStr([
  14628. '']));
  14629. end;
  14630. procedure TTestModule.TestClass_LocalConstDuplicate_Unit;
  14631. begin
  14632. StartUnit(false);
  14633. Add([
  14634. 'interface',
  14635. 'type',
  14636. ' TObject = class',
  14637. ' const cI: longint = 3;',
  14638. ' procedure Fly;',
  14639. ' procedure Run;',
  14640. ' end;',
  14641. ' TBird = class',
  14642. ' procedure Go;',
  14643. ' end;',
  14644. 'implementation',
  14645. 'procedure tobject.fly;',
  14646. 'const cI: word = 4;',
  14647. 'begin',
  14648. ' if cI=Self.cI then ;',
  14649. 'end;',
  14650. 'procedure tobject.run;',
  14651. 'const cI: word = 5;',
  14652. 'begin',
  14653. ' if cI=Self.cI then ;',
  14654. 'end;',
  14655. 'procedure tbird.go;',
  14656. 'const cI: word = 6;',
  14657. 'begin',
  14658. ' if cI=Self.cI then ;',
  14659. 'end;',
  14660. '']);
  14661. ConvertUnit;
  14662. CheckSource('TestClass_LocalConstDuplicate_Unit',
  14663. LinesToStr([
  14664. 'rtl.createClass(this, "TObject", null, function () {',
  14665. ' this.cI = 3;',
  14666. ' this.$init = function () {',
  14667. ' };',
  14668. ' this.$final = function () {',
  14669. ' };',
  14670. ' var cI$1 = 4;',
  14671. ' this.Fly = function () {',
  14672. ' if (cI$1 === this.cI) ;',
  14673. ' };',
  14674. ' var cI$2 = 5;',
  14675. ' this.Run = function () {',
  14676. ' if (cI$2 === this.cI) ;',
  14677. ' };',
  14678. '});',
  14679. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  14680. ' var cI$3 = 6;',
  14681. ' this.Go = function () {',
  14682. ' if (cI$3 === this.cI) ;',
  14683. ' };',
  14684. '});',
  14685. '']),
  14686. '',
  14687. '');
  14688. end;
  14689. procedure TTestModule.TestClass_LocalVarSelfFail;
  14690. begin
  14691. StartProgram(false);
  14692. Add([
  14693. 'type',
  14694. ' TObject = class',
  14695. ' constructor Create;',
  14696. ' end;',
  14697. 'constructor tobject.create;',
  14698. 'var self: longint;',
  14699. 'begin',
  14700. 'end',
  14701. 'begin',
  14702. '']);
  14703. SetExpectedPasResolverError('Duplicate identifier "self" at (0)',nDuplicateIdentifier);
  14704. ConvertProgram;
  14705. end;
  14706. procedure TTestModule.TestClass_ArgSelfFail;
  14707. begin
  14708. StartProgram(false);
  14709. Add([
  14710. 'type',
  14711. ' TObject = class',
  14712. ' procedure DoIt(Self: longint);',
  14713. ' end;',
  14714. 'procedure tobject.doit(self: longint);',
  14715. 'begin',
  14716. 'end',
  14717. 'begin',
  14718. '']);
  14719. SetExpectedPasResolverError('Duplicate identifier "Self" at test1.pp(5,24)',nDuplicateIdentifier);
  14720. ConvertProgram;
  14721. end;
  14722. procedure TTestModule.TestClass_NestedProcSelf;
  14723. begin
  14724. StartProgram(false);
  14725. Add([
  14726. 'type',
  14727. ' TObject = class',
  14728. ' Key: longint;',
  14729. ' class var State: longint;',
  14730. ' procedure DoIt;',
  14731. ' function GetSize: longint; virtual; abstract;',
  14732. ' procedure SetSize(Value: longint); virtual; abstract;',
  14733. ' property Size: longint read GetSize write SetSize;',
  14734. ' end;',
  14735. 'procedure tobject.doit;',
  14736. ' procedure Sub;',
  14737. ' begin',
  14738. ' key:=key+2;',
  14739. ' self.key:=self.key+3;',
  14740. ' state:=state+4;',
  14741. ' self.state:=self.state+5;',
  14742. ' tobject.state:=tobject.state+6;',
  14743. ' size:=size+7;',
  14744. ' self.size:=self.size+8;',
  14745. ' end;',
  14746. 'begin',
  14747. ' sub;',
  14748. ' key:=key+12;',
  14749. ' self.key:=self.key+13;',
  14750. ' state:=state+14;',
  14751. ' self.state:=self.state+15;',
  14752. ' tobject.state:=tobject.state+16;',
  14753. ' size:=size+17;',
  14754. ' self.size:=self.size+18;',
  14755. 'end;',
  14756. 'begin',
  14757. '']);
  14758. ConvertProgram;
  14759. CheckSource('TestClass_NestedProcSelf',
  14760. LinesToStr([ // statements
  14761. 'rtl.createClass(this, "TObject", null, function () {',
  14762. ' this.State = 0;',
  14763. ' this.$init = function () {',
  14764. ' this.Key = 0;',
  14765. ' };',
  14766. ' this.$final = function () {',
  14767. ' };',
  14768. ' this.DoIt = function () {',
  14769. ' var $Self = this;',
  14770. ' function Sub() {',
  14771. ' $Self.Key = $Self.Key + 2;',
  14772. ' $Self.Key = $Self.Key + 3;',
  14773. ' $mod.TObject.State = $Self.State + 4;',
  14774. ' $mod.TObject.State = $Self.State + 5;',
  14775. ' $mod.TObject.State = $mod.TObject.State + 6;',
  14776. ' $Self.SetSize($Self.GetSize() + 7);',
  14777. ' $Self.SetSize($Self.GetSize() + 8);',
  14778. ' };',
  14779. ' Sub();',
  14780. ' this.Key = this.Key + 12;',
  14781. ' $Self.Key = $Self.Key + 13;',
  14782. ' $mod.TObject.State = this.State + 14;',
  14783. ' $mod.TObject.State = $Self.State + 15;',
  14784. ' $mod.TObject.State = $mod.TObject.State + 16;',
  14785. ' this.SetSize(this.GetSize() + 17);',
  14786. ' $Self.SetSize($Self.GetSize() + 18);',
  14787. ' };',
  14788. '});',
  14789. '']),
  14790. LinesToStr([ // $mod.$main
  14791. '']));
  14792. end;
  14793. procedure TTestModule.TestClass_NestedProcSelf2;
  14794. begin
  14795. StartProgram(false);
  14796. Add([
  14797. 'type',
  14798. ' TObject = class',
  14799. ' Key: longint;',
  14800. ' class var State: longint;',
  14801. ' function GetSize: longint; virtual; abstract;',
  14802. ' procedure SetSize(Value: longint); virtual; abstract;',
  14803. ' property Size: longint read GetSize write SetSize;',
  14804. ' end;',
  14805. ' TBird = class',
  14806. ' procedure DoIt;',
  14807. ' end;',
  14808. 'procedure tbird.doit;',
  14809. ' procedure Sub;',
  14810. ' begin',
  14811. ' key:=key+2;',
  14812. ' self.key:=self.key+3;',
  14813. ' state:=state+4;',
  14814. ' self.state:=self.state+5;',
  14815. ' tobject.state:=tobject.state+6;',
  14816. ' size:=size+7;',
  14817. ' self.size:=self.size+8;',
  14818. ' end;',
  14819. 'begin',
  14820. ' sub;',
  14821. ' key:=key+12;',
  14822. ' self.key:=self.key+13;',
  14823. ' state:=state+14;',
  14824. ' self.state:=self.state+15;',
  14825. ' tobject.state:=tobject.state+16;',
  14826. ' size:=size+17;',
  14827. ' self.size:=self.size+18;',
  14828. 'end;',
  14829. 'begin',
  14830. '']);
  14831. ConvertProgram;
  14832. CheckSource('TestClass_NestedProcSelf2',
  14833. LinesToStr([ // statements
  14834. 'rtl.createClass(this, "TObject", null, function () {',
  14835. ' this.State = 0;',
  14836. ' this.$init = function () {',
  14837. ' this.Key = 0;',
  14838. ' };',
  14839. ' this.$final = function () {',
  14840. ' };',
  14841. '});',
  14842. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  14843. ' this.DoIt = function () {',
  14844. ' var $Self = this;',
  14845. ' function Sub() {',
  14846. ' $Self.Key = $Self.Key + 2;',
  14847. ' $Self.Key = $Self.Key + 3;',
  14848. ' $mod.TObject.State = $Self.State + 4;',
  14849. ' $mod.TObject.State = $Self.State + 5;',
  14850. ' $mod.TObject.State = $mod.TObject.State + 6;',
  14851. ' $Self.SetSize($Self.GetSize() + 7);',
  14852. ' $Self.SetSize($Self.GetSize() + 8);',
  14853. ' };',
  14854. ' Sub();',
  14855. ' this.Key = this.Key + 12;',
  14856. ' $Self.Key = $Self.Key + 13;',
  14857. ' $mod.TObject.State = this.State + 14;',
  14858. ' $mod.TObject.State = $Self.State + 15;',
  14859. ' $mod.TObject.State = $mod.TObject.State + 16;',
  14860. ' this.SetSize(this.GetSize() + 17);',
  14861. ' $Self.SetSize($Self.GetSize() + 18);',
  14862. ' };',
  14863. '});',
  14864. '']),
  14865. LinesToStr([ // $mod.$main
  14866. '']));
  14867. end;
  14868. procedure TTestModule.TestClass_NestedProcClassSelf;
  14869. begin
  14870. StartProgram(false);
  14871. Add([
  14872. 'type',
  14873. ' TObject = class',
  14874. ' class var State: longint;',
  14875. ' class procedure DoIt;',
  14876. ' class function GetSize: longint; virtual; abstract;',
  14877. ' class procedure SetSize(Value: longint); virtual; abstract;',
  14878. ' class property Size: longint read GetSize write SetSize;',
  14879. ' end;',
  14880. 'class procedure tobject.doit;',
  14881. ' procedure Sub;',
  14882. ' begin',
  14883. ' state:=state+2;',
  14884. ' self.state:=self.state+3;',
  14885. ' tobject.state:=tobject.state+4;',
  14886. ' size:=size+5;',
  14887. ' self.size:=self.size+6;',
  14888. ' tobject.size:=tobject.size+7;',
  14889. ' end;',
  14890. 'begin',
  14891. ' sub;',
  14892. ' state:=state+12;',
  14893. ' self.state:=self.state+13;',
  14894. ' tobject.state:=tobject.state+14;',
  14895. ' size:=size+15;',
  14896. ' self.size:=self.size+16;',
  14897. ' tobject.size:=tobject.size+17;',
  14898. 'end;',
  14899. 'begin',
  14900. '']);
  14901. ConvertProgram;
  14902. CheckSource('TestClass_NestedProcClassSelf',
  14903. LinesToStr([ // statements
  14904. 'rtl.createClass(this, "TObject", null, function () {',
  14905. ' this.State = 0;',
  14906. ' this.$init = function () {',
  14907. ' };',
  14908. ' this.$final = function () {',
  14909. ' };',
  14910. ' this.DoIt = function () {',
  14911. ' var $Self = this;',
  14912. ' function Sub() {',
  14913. ' $mod.TObject.State = $Self.State + 2;',
  14914. ' $mod.TObject.State = $Self.State + 3;',
  14915. ' $mod.TObject.State = $mod.TObject.State + 4;',
  14916. ' $Self.SetSize($Self.GetSize() + 5);',
  14917. ' $Self.SetSize($Self.GetSize() + 6);',
  14918. ' $mod.TObject.SetSize($mod.TObject.GetSize() + 7);',
  14919. ' };',
  14920. ' Sub();',
  14921. ' $mod.TObject.State = this.State + 12;',
  14922. ' $mod.TObject.State = $Self.State + 13;',
  14923. ' $mod.TObject.State = $mod.TObject.State + 14;',
  14924. ' this.SetSize(this.GetSize() + 15);',
  14925. ' $Self.SetSize($Self.GetSize() + 16);',
  14926. ' $mod.TObject.SetSize($mod.TObject.GetSize() + 17);',
  14927. ' };',
  14928. '});',
  14929. '']),
  14930. LinesToStr([ // $mod.$main
  14931. '']));
  14932. end;
  14933. procedure TTestModule.TestClass_NestedProcCallInherited;
  14934. begin
  14935. StartProgram(false);
  14936. Add([
  14937. 'type',
  14938. ' TObject = class',
  14939. ' function DoIt(k: boolean): longint; virtual;',
  14940. ' end;',
  14941. ' TBird = class',
  14942. ' function DoIt(k: boolean): longint; override;',
  14943. ' end;',
  14944. 'function tobject.doit(k: boolean): longint;',
  14945. 'begin',
  14946. 'end;',
  14947. 'function tbird.doit(k: boolean): longint;',
  14948. ' procedure Sub;',
  14949. ' begin',
  14950. ' inherited DoIt(true);',
  14951. //' if inherited DoIt(false)=4 then ;',
  14952. ' end;',
  14953. 'begin',
  14954. ' Sub;',
  14955. ' inherited;',
  14956. ' inherited DoIt(true);',
  14957. //' if inherited DoIt(false)=14 then ;',
  14958. 'end;',
  14959. 'begin',
  14960. '']);
  14961. ConvertProgram;
  14962. CheckSource('TestClass_NestedProcCallInherited',
  14963. LinesToStr([ // statements
  14964. 'rtl.createClass(this, "TObject", null, function () {',
  14965. ' this.$init = function () {',
  14966. ' };',
  14967. ' this.$final = function () {',
  14968. ' };',
  14969. ' this.DoIt = function (k) {',
  14970. ' var Result = 0;',
  14971. ' return Result;',
  14972. ' };',
  14973. '});',
  14974. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  14975. ' this.DoIt = function (k) {',
  14976. ' var $Self = this;',
  14977. ' var Result = 0;',
  14978. ' function Sub() {',
  14979. ' $mod.TObject.DoIt.call($Self, true);',
  14980. ' };',
  14981. ' Sub();',
  14982. ' $mod.TObject.DoIt.apply(this, arguments);',
  14983. ' $mod.TObject.DoIt.call(this, true);',
  14984. ' return Result;',
  14985. ' };',
  14986. '});',
  14987. '']),
  14988. LinesToStr([ // $mod.$main
  14989. '']));
  14990. end;
  14991. procedure TTestModule.TestClass_TObjectFree;
  14992. begin
  14993. StartProgram(false);
  14994. Add([
  14995. 'type',
  14996. ' TObject = class',
  14997. ' Obj: tobject;',
  14998. ' procedure Free;',
  14999. ' procedure Release;',
  15000. ' end;',
  15001. 'procedure tobject.free;',
  15002. 'begin',
  15003. 'end;',
  15004. 'procedure tobject.release;',
  15005. 'begin',
  15006. ' free;',
  15007. ' if true then free;',
  15008. 'end;',
  15009. 'function DoIt(o: tobject): tobject;',
  15010. 'var l: tobject;',
  15011. 'begin',
  15012. ' o.free;',
  15013. ' o.free();',
  15014. ' l.free;',
  15015. ' l.free();',
  15016. ' o.obj.free;',
  15017. ' o.obj.free();',
  15018. ' with o do obj.free;',
  15019. ' with o do obj.free();',
  15020. ' result.Free;',
  15021. ' result.Free();',
  15022. 'end;',
  15023. 'var o: tobject;',
  15024. ' a: array of tobject;',
  15025. 'begin',
  15026. ' o.free;',
  15027. ' o.obj.free;',
  15028. ' a[1+2].free;',
  15029. '']);
  15030. ConvertProgram;
  15031. CheckSource('TestClass_TObjectFree',
  15032. LinesToStr([ // statements
  15033. 'rtl.createClass(this, "TObject", null, function () {',
  15034. ' this.$init = function () {',
  15035. ' this.Obj = null;',
  15036. ' };',
  15037. ' this.$final = function () {',
  15038. ' this.Obj = undefined;',
  15039. ' };',
  15040. ' this.Free = function () {',
  15041. ' };',
  15042. ' this.Release = function () {',
  15043. ' this.Free();',
  15044. ' if (true) this.Free();',
  15045. ' };',
  15046. '});',
  15047. 'this.DoIt = function (o) {',
  15048. ' var Result = null;',
  15049. ' var l = null;',
  15050. ' o = rtl.freeLoc(o);',
  15051. ' o = rtl.freeLoc(o);',
  15052. ' l = rtl.freeLoc(l);',
  15053. ' l = rtl.freeLoc(l);',
  15054. ' rtl.free(o, "Obj");',
  15055. ' rtl.free(o, "Obj");',
  15056. ' rtl.free(o, "Obj");',
  15057. ' rtl.free(o, "Obj");',
  15058. ' Result = rtl.freeLoc(Result);',
  15059. ' Result = rtl.freeLoc(Result);',
  15060. ' return Result;',
  15061. '};',
  15062. 'this.o = null;',
  15063. 'this.a = [];',
  15064. '']),
  15065. LinesToStr([ // $mod.$main
  15066. 'rtl.free($mod, "o");',
  15067. 'rtl.free($mod.o, "Obj");',
  15068. 'rtl.free($mod.a, 1 + 2);',
  15069. '']));
  15070. end;
  15071. procedure TTestModule.TestClass_TObjectFree_VarArg;
  15072. begin
  15073. StartProgram(false);
  15074. Add([
  15075. 'type',
  15076. ' TObject = class',
  15077. ' Obj: tobject;',
  15078. ' procedure Free;',
  15079. ' end;',
  15080. 'procedure tobject.free;',
  15081. 'begin',
  15082. 'end;',
  15083. 'procedure DoIt(var o: tobject);',
  15084. 'begin',
  15085. ' o.free;',
  15086. ' o.free();',
  15087. 'end;',
  15088. 'begin',
  15089. '']);
  15090. ConvertProgram;
  15091. CheckSource('TestClass_TObjectFree_VarArg',
  15092. LinesToStr([ // statements
  15093. 'rtl.createClass(this, "TObject", null, function () {',
  15094. ' this.$init = function () {',
  15095. ' this.Obj = null;',
  15096. ' };',
  15097. ' this.$final = function () {',
  15098. ' this.Obj = undefined;',
  15099. ' };',
  15100. ' this.Free = function () {',
  15101. ' };',
  15102. '});',
  15103. 'this.DoIt = function (o) {',
  15104. ' o.set(rtl.freeLoc(o.get()));',
  15105. ' o.set(rtl.freeLoc(o.get()));',
  15106. '};',
  15107. '']),
  15108. LinesToStr([ // $mod.$main
  15109. '']));
  15110. end;
  15111. procedure TTestModule.TestClass_TObjectFreeNewInstance;
  15112. begin
  15113. StartProgram(false);
  15114. Add([
  15115. 'type',
  15116. ' TObject = class',
  15117. ' constructor Create;',
  15118. ' procedure Free;',
  15119. ' end;',
  15120. 'constructor TObject.Create; begin end;',
  15121. 'procedure tobject.free; begin end;',
  15122. 'begin',
  15123. ' with tobject.create do free;',
  15124. '']);
  15125. ConvertProgram;
  15126. CheckSource('TestClass_TObjectFreeNewInstance',
  15127. LinesToStr([ // statements
  15128. 'rtl.createClass(this, "TObject", null, function () {',
  15129. ' this.$init = function () {',
  15130. ' };',
  15131. ' this.$final = function () {',
  15132. ' };',
  15133. ' this.Create = function () {',
  15134. ' return this;',
  15135. ' };',
  15136. ' this.Free = function () {',
  15137. ' };',
  15138. '});',
  15139. '']),
  15140. LinesToStr([ // $mod.$main
  15141. 'var $with = $mod.TObject.$create("Create");',
  15142. '$with=rtl.freeLoc($with);',
  15143. '']));
  15144. end;
  15145. procedure TTestModule.TestClass_TObjectFreeLowerCase;
  15146. begin
  15147. StartProgram(false);
  15148. Add([
  15149. 'type',
  15150. ' TObject = class',
  15151. ' destructor Destroy;',
  15152. ' procedure Free;',
  15153. ' end;',
  15154. 'destructor TObject.Destroy; begin end;',
  15155. 'procedure tobject.free; begin end;',
  15156. 'var o: tobject;',
  15157. 'begin',
  15158. ' o.free;',
  15159. '']);
  15160. Converter.UseLowerCase:=true;
  15161. ConvertProgram;
  15162. CheckSource('TestClass_TObjectFreeLowerCase',
  15163. LinesToStr([ // statements
  15164. 'rtl.createClass(this, "tobject", null, function () {',
  15165. ' this.$init = function () {',
  15166. ' };',
  15167. ' this.$final = function () {',
  15168. ' };',
  15169. ' rtl.tObjectDestroy = "destroy";',
  15170. ' this.destroy = function () {',
  15171. ' };',
  15172. ' this.free = function () {',
  15173. ' };',
  15174. '});',
  15175. 'this.o = null;',
  15176. '']),
  15177. LinesToStr([ // $mod.$main
  15178. 'rtl.free($mod, "o");',
  15179. '']));
  15180. end;
  15181. procedure TTestModule.TestClass_TObjectFreeFunctionFail;
  15182. begin
  15183. StartProgram(false);
  15184. Add([
  15185. 'type',
  15186. ' TObject = class',
  15187. ' procedure Free;',
  15188. ' function GetObj: tobject; virtual; abstract;',
  15189. ' end;',
  15190. 'procedure tobject.free;',
  15191. 'begin',
  15192. 'end;',
  15193. 'var o: tobject;',
  15194. 'begin',
  15195. ' o.getobj.free;',
  15196. '']);
  15197. SetExpectedPasResolverError(sFreeNeedsVar,nFreeNeedsVar);
  15198. ConvertProgram;
  15199. end;
  15200. procedure TTestModule.TestClass_TObjectFreePropertyFail;
  15201. begin
  15202. StartProgram(false);
  15203. Add([
  15204. 'type',
  15205. ' TObject = class',
  15206. ' procedure Free;',
  15207. ' FObj: TObject;',
  15208. ' property Obj: tobject read FObj write FObj;',
  15209. ' end;',
  15210. 'procedure tobject.free;',
  15211. 'begin',
  15212. 'end;',
  15213. 'var o: tobject;',
  15214. 'begin',
  15215. ' o.obj.free;',
  15216. '']);
  15217. SetExpectedPasResolverError(sFreeNeedsVar,nFreeNeedsVar);
  15218. ConvertProgram;
  15219. end;
  15220. procedure TTestModule.TestClass_ForIn;
  15221. begin
  15222. StartProgram(false);
  15223. Add([
  15224. 'type',
  15225. ' TObject = class end;',
  15226. ' TItem = TObject;',
  15227. ' TEnumerator = class',
  15228. ' FCurrent: TItem;',
  15229. ' property Current: TItem read FCurrent;',
  15230. ' function MoveNext: boolean;',
  15231. ' end;',
  15232. ' TBird = class',
  15233. ' function GetEnumerator: TEnumerator;',
  15234. ' end;',
  15235. 'function TEnumerator.MoveNext: boolean;',
  15236. 'begin',
  15237. 'end;',
  15238. 'function TBird.GetEnumerator: TEnumerator;',
  15239. 'begin',
  15240. 'end;',
  15241. 'var',
  15242. ' b: TBird;',
  15243. ' i, i2: TItem;',
  15244. 'begin',
  15245. ' for i in b do i2:=i;']);
  15246. ConvertProgram;
  15247. CheckSource('TestClass_ForIn',
  15248. LinesToStr([ // statements
  15249. 'rtl.createClass(this, "TObject", null, function () {',
  15250. ' this.$init = function () {',
  15251. ' };',
  15252. ' this.$final = function () {',
  15253. ' };',
  15254. '});',
  15255. 'rtl.createClass(this, "TEnumerator", this.TObject, function () {',
  15256. ' this.$init = function () {',
  15257. ' $mod.TObject.$init.call(this);',
  15258. ' this.FCurrent = null;',
  15259. ' };',
  15260. ' this.$final = function () {',
  15261. ' this.FCurrent = undefined;',
  15262. ' $mod.TObject.$final.call(this);',
  15263. ' };',
  15264. ' this.MoveNext = function () {',
  15265. ' var Result = false;',
  15266. ' return Result;',
  15267. ' };',
  15268. '});',
  15269. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  15270. ' this.GetEnumerator = function () {',
  15271. ' var Result = null;',
  15272. ' return Result;',
  15273. ' };',
  15274. '});',
  15275. 'this.b = null;',
  15276. 'this.i = null;',
  15277. 'this.i2 = null;'
  15278. ]),
  15279. LinesToStr([ // $mod.$main
  15280. 'var $in = $mod.b.GetEnumerator();',
  15281. 'try {',
  15282. ' while ($in.MoveNext()){',
  15283. ' $mod.i = $in.FCurrent;',
  15284. ' $mod.i2 = $mod.i;',
  15285. ' }',
  15286. '} finally {',
  15287. ' $in = rtl.freeLoc($in)',
  15288. '};',
  15289. '']));
  15290. end;
  15291. procedure TTestModule.TestClass_DispatchMessage;
  15292. begin
  15293. StartProgram(false);
  15294. Add([
  15295. 'type',
  15296. ' TObject = class',
  15297. ' {$DispatchField DispInt}',
  15298. ' procedure Dispatch(var Msg); virtual; abstract;',
  15299. ' {$DispatchStrField DispStr}',
  15300. ' procedure DispatchStr(var Msg); virtual; abstract;',
  15301. ' end;',
  15302. ' THopMsg = record',
  15303. ' DispInt: longint;',
  15304. ' end;',
  15305. ' TPutMsg = record',
  15306. ' DispStr: string;',
  15307. ' end;',
  15308. ' TBird = class',
  15309. ' procedure Fly(var Msg); virtual; abstract; message 2;',
  15310. ' procedure Run; overload; virtual; abstract;',
  15311. ' procedure Run(var Msg); overload; message ''Fast'';',
  15312. ' procedure Hop(var Msg: THopMsg); virtual; abstract; message 3;',
  15313. ' procedure Put(var Msg: TPutMsg); virtual; abstract; message ''foo'';',
  15314. ' end;',
  15315. 'procedure TBird.Run(var Msg);',
  15316. 'begin',
  15317. 'end;',
  15318. 'begin',
  15319. '']);
  15320. ConvertProgram;
  15321. CheckSource('TestClass_Message',
  15322. LinesToStr([ // statements
  15323. 'rtl.createClass(this, "TObject", null, function () {',
  15324. ' this.$init = function () {',
  15325. ' };',
  15326. ' this.$final = function () {',
  15327. ' };',
  15328. '});',
  15329. 'rtl.recNewT(this, "THopMsg", function () {',
  15330. ' this.DispInt = 0;',
  15331. ' this.$eq = function (b) {',
  15332. ' return this.DispInt === b.DispInt;',
  15333. ' };',
  15334. ' this.$assign = function (s) {',
  15335. ' this.DispInt = s.DispInt;',
  15336. ' return this;',
  15337. ' };',
  15338. '});',
  15339. 'rtl.recNewT(this, "TPutMsg", function () {',
  15340. ' this.DispStr = "";',
  15341. ' this.$eq = function (b) {',
  15342. ' return this.DispStr === b.DispStr;',
  15343. ' };',
  15344. ' this.$assign = function (s) {',
  15345. ' this.DispStr = s.DispStr;',
  15346. ' return this;',
  15347. ' };',
  15348. '});',
  15349. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  15350. ' this.Run$1 = function (Msg) {',
  15351. ' };',
  15352. ' this.$msgint = {',
  15353. ' "2": "Fly",',
  15354. ' "3": "Hop"',
  15355. ' };',
  15356. ' this.$msgstr = {',
  15357. ' Fast: "Run$1",',
  15358. ' foo: "Put"',
  15359. ' };',
  15360. '});',
  15361. '']),
  15362. LinesToStr([ // $mod.$main
  15363. '']));
  15364. end;
  15365. procedure TTestModule.TestClass_Message_DuplicateIntFail;
  15366. begin
  15367. StartProgram(false);
  15368. Add([
  15369. 'type',
  15370. ' TObject = class',
  15371. ' procedure Fly(var Msg); virtual; abstract; message 3;',
  15372. ' procedure Run(var Msg); virtual; abstract; message 1+2;',
  15373. ' end;',
  15374. 'begin',
  15375. '']);
  15376. SetExpectedPasResolverError('Duplicate message id "3" at test1.pp(5,56)',nDuplicateMessageIdXAtY);
  15377. ConvertProgram;
  15378. end;
  15379. procedure TTestModule.TestClass_DispatchMessage_WrongFieldNameFail;
  15380. begin
  15381. StartProgram(false);
  15382. Add([
  15383. 'type',
  15384. ' TObject = class',
  15385. ' {$dispatchfield Msg}',
  15386. ' procedure Dispatch(var Msg); virtual; abstract;',
  15387. ' end;',
  15388. ' TFlyMsg = record',
  15389. ' FlyId: longint;',
  15390. ' end;',
  15391. ' TBird = class',
  15392. ' procedure Fly(var Msg: TFlyMsg); virtual; abstract; message 3;',
  15393. ' end;',
  15394. 'begin',
  15395. '']);
  15396. ConvertProgram;
  15397. CheckHint(mtWarning,nDispatchRequiresX,'Dispatch requires record field "Msg"');
  15398. end;
  15399. procedure TTestModule.TestClassOf_Create;
  15400. begin
  15401. StartProgram(false);
  15402. Add('type');
  15403. Add(' TObject = class');
  15404. Add(' constructor Create;');
  15405. Add(' end;');
  15406. Add(' TClass = class of TObject;');
  15407. Add('constructor tobject.create; begin end;');
  15408. Add('var');
  15409. Add(' Obj: tobject;');
  15410. Add(' C: tclass;');
  15411. Add('begin');
  15412. Add(' obj:=C.create;');
  15413. Add(' with c do obj:=create;');
  15414. ConvertProgram;
  15415. CheckSource('TestClassOf_Create',
  15416. LinesToStr([ // statements
  15417. 'rtl.createClass(this, "TObject", null, function () {',
  15418. ' this.$init = function () {',
  15419. ' };',
  15420. ' this.$final = function () {',
  15421. ' };',
  15422. ' this.Create = function () {',
  15423. ' return this;',
  15424. ' };',
  15425. '});',
  15426. 'this.Obj = null;',
  15427. 'this.C = null;'
  15428. ]),
  15429. LinesToStr([ // $mod.$main
  15430. '$mod.Obj = $mod.C.$create("Create");',
  15431. 'var $with = $mod.C;',
  15432. '$mod.Obj = $with.$create("Create");',
  15433. '']));
  15434. end;
  15435. procedure TTestModule.TestClassOf_Call;
  15436. begin
  15437. StartProgram(false);
  15438. Add('type');
  15439. Add(' TObject = class');
  15440. Add(' class procedure DoIt;');
  15441. Add(' end;');
  15442. Add(' TClass = class of TObject;');
  15443. Add('class procedure tobject.doit; begin end;');
  15444. Add('var');
  15445. Add(' C: tclass;');
  15446. Add('begin');
  15447. Add(' c.doit;');
  15448. Add(' with c do doit;');
  15449. ConvertProgram;
  15450. CheckSource('TestClassOf_Call',
  15451. LinesToStr([ // statements
  15452. 'rtl.createClass(this, "TObject", null, function () {',
  15453. ' this.$init = function () {',
  15454. ' };',
  15455. ' this.$final = function () {',
  15456. ' };',
  15457. ' this.DoIt = function () {',
  15458. ' };',
  15459. '});',
  15460. 'this.C = null;'
  15461. ]),
  15462. LinesToStr([ // $mod.$main
  15463. '$mod.C.DoIt();',
  15464. 'var $with = $mod.C;',
  15465. '$with.DoIt();',
  15466. '']));
  15467. end;
  15468. procedure TTestModule.TestClassOf_Assign;
  15469. begin
  15470. StartProgram(false);
  15471. Add('type');
  15472. Add(' TClass = class of TObject;');
  15473. Add(' TObject = class');
  15474. Add(' ClassType: TClass; ');
  15475. Add(' end;');
  15476. Add('var');
  15477. Add(' Obj: tobject;');
  15478. Add(' C: tclass;');
  15479. Add('begin');
  15480. Add(' c:=nil;');
  15481. Add(' c:=obj.classtype;');
  15482. ConvertProgram;
  15483. CheckSource('TestClassOf_Assign',
  15484. LinesToStr([ // statements
  15485. 'rtl.createClass(this, "TObject", null, function () {',
  15486. ' this.$init = function () {',
  15487. ' this.ClassType = null;',
  15488. ' };',
  15489. ' this.$final = function () {',
  15490. ' this.ClassType = undefined;',
  15491. ' };',
  15492. '});',
  15493. 'this.Obj = null;',
  15494. 'this.C = null;'
  15495. ]),
  15496. LinesToStr([ // $mod.$main
  15497. '$mod.C = null;',
  15498. '$mod.C = $mod.Obj.ClassType;',
  15499. '']));
  15500. end;
  15501. procedure TTestModule.TestClassOf_Is;
  15502. begin
  15503. StartProgram(false);
  15504. Add('type');
  15505. Add(' TClass = class of TObject;');
  15506. Add(' TObject = class');
  15507. Add(' end;');
  15508. Add(' TCar = class');
  15509. Add(' end;');
  15510. Add(' TCars = class of TCar;');
  15511. Add('var');
  15512. Add(' Obj: tobject;');
  15513. Add(' C: tclass;');
  15514. Add(' Cars: tcars;');
  15515. Add('begin');
  15516. Add(' if c is tcar then ;');
  15517. Add(' if c is tcars then ;');
  15518. ConvertProgram;
  15519. CheckSource('TestClassOf_Is',
  15520. LinesToStr([ // statements
  15521. 'rtl.createClass(this, "TObject", null, function () {',
  15522. ' this.$init = function () {',
  15523. ' };',
  15524. ' this.$final = function () {',
  15525. ' };',
  15526. '});',
  15527. 'rtl.createClass(this, "TCar", this.TObject, function () {',
  15528. '});',
  15529. 'this.Obj = null;',
  15530. 'this.C = null;',
  15531. 'this.Cars = null;'
  15532. ]),
  15533. LinesToStr([ // $mod.$main
  15534. 'if(rtl.is($mod.C,$mod.TCar));',
  15535. 'if(rtl.is($mod.C,$mod.TCar));',
  15536. '']));
  15537. end;
  15538. procedure TTestModule.TestClassOf_Compare;
  15539. begin
  15540. StartProgram(false);
  15541. Add('type');
  15542. Add(' TClass = class of TObject;');
  15543. Add(' TObject = class');
  15544. Add(' ClassType: TClass; ');
  15545. Add(' end;');
  15546. Add('var');
  15547. Add(' b: boolean;');
  15548. Add(' Obj: tobject;');
  15549. Add(' C: tclass;');
  15550. Add('begin');
  15551. Add(' b:=c=nil;');
  15552. Add(' b:=nil=c;');
  15553. Add(' b:=c=obj.classtype;');
  15554. Add(' b:=obj.classtype=c;');
  15555. Add(' b:=c=TObject;');
  15556. Add(' b:=TObject=c;');
  15557. Add(' b:=c<>nil;');
  15558. Add(' b:=nil<>c;');
  15559. Add(' b:=c<>obj.classtype;');
  15560. Add(' b:=obj.classtype<>c;');
  15561. Add(' b:=c<>TObject;');
  15562. Add(' b:=TObject<>c;');
  15563. ConvertProgram;
  15564. CheckSource('TestClassOf_Compare',
  15565. LinesToStr([ // statements
  15566. 'rtl.createClass(this, "TObject", null, function () {',
  15567. ' this.$init = function () {',
  15568. ' this.ClassType = null;',
  15569. ' };',
  15570. ' this.$final = function () {',
  15571. ' this.ClassType = undefined;',
  15572. ' };',
  15573. '});',
  15574. 'this.b = false;',
  15575. 'this.Obj = null;',
  15576. 'this.C = null;'
  15577. ]),
  15578. LinesToStr([ // $mod.$main
  15579. '$mod.b = $mod.C === null;',
  15580. '$mod.b = null === $mod.C;',
  15581. '$mod.b = $mod.C === $mod.Obj.ClassType;',
  15582. '$mod.b = $mod.Obj.ClassType === $mod.C;',
  15583. '$mod.b = $mod.C === $mod.TObject;',
  15584. '$mod.b = $mod.TObject === $mod.C;',
  15585. '$mod.b = $mod.C !== null;',
  15586. '$mod.b = null !== $mod.C;',
  15587. '$mod.b = $mod.C !== $mod.Obj.ClassType;',
  15588. '$mod.b = $mod.Obj.ClassType !== $mod.C;',
  15589. '$mod.b = $mod.C !== $mod.TObject;',
  15590. '$mod.b = $mod.TObject !== $mod.C;',
  15591. '']));
  15592. end;
  15593. procedure TTestModule.TestClassOf_ClassVar;
  15594. begin
  15595. StartProgram(false);
  15596. Add('type');
  15597. Add(' TObject = class');
  15598. Add(' class var id: longint;');
  15599. Add(' end;');
  15600. Add(' TClass = class of TObject;');
  15601. Add('var');
  15602. Add(' C: tclass;');
  15603. Add('begin');
  15604. Add(' C.id:=C.id;');
  15605. ConvertProgram;
  15606. CheckSource('TestClassOf_ClassVar',
  15607. LinesToStr([ // statements
  15608. 'rtl.createClass(this, "TObject", null, function () {',
  15609. ' this.id = 0;',
  15610. ' this.$init = function () {',
  15611. ' };',
  15612. ' this.$final = function () {',
  15613. ' };',
  15614. '});',
  15615. 'this.C = null;'
  15616. ]),
  15617. LinesToStr([ // $mod.$main
  15618. '$mod.TObject.id = $mod.C.id;',
  15619. '']));
  15620. end;
  15621. procedure TTestModule.TestClassOf_ClassMethod;
  15622. begin
  15623. StartProgram(false);
  15624. Add('type');
  15625. Add(' TObject = class');
  15626. Add(' class function DoIt(i: longint = 0): longint;');
  15627. Add(' end;');
  15628. Add(' TClass = class of TObject;');
  15629. Add('class function tobject.doit(i: longint = 0): longint; begin end;');
  15630. Add('var');
  15631. Add(' i: longint;');
  15632. Add(' C: tclass;');
  15633. Add('begin');
  15634. Add(' C.DoIt;');
  15635. Add(' C.DoIt();');
  15636. Add(' i:=C.DoIt;');
  15637. Add(' i:=C.DoIt();');
  15638. ConvertProgram;
  15639. CheckSource('TestClassOf_ClassMethod',
  15640. LinesToStr([ // statements
  15641. 'rtl.createClass(this, "TObject", null, function () {',
  15642. ' this.$init = function () {',
  15643. ' };',
  15644. ' this.$final = function () {',
  15645. ' };',
  15646. ' this.DoIt = function (i) {',
  15647. ' var Result = 0;',
  15648. ' return Result;',
  15649. ' };',
  15650. '});',
  15651. 'this.i = 0;',
  15652. 'this.C = null;'
  15653. ]),
  15654. LinesToStr([ // $mod.$main
  15655. '$mod.C.DoIt(0);',
  15656. '$mod.C.DoIt(0);',
  15657. '$mod.i = $mod.C.DoIt(0);',
  15658. '$mod.i = $mod.C.DoIt(0);',
  15659. '']));
  15660. end;
  15661. procedure TTestModule.TestClassOf_ClassProperty;
  15662. begin
  15663. StartProgram(false);
  15664. Add([
  15665. 'type',
  15666. ' TObject = class',
  15667. ' class var FA: longint;',
  15668. ' class function GetA: longint;',
  15669. ' class procedure SetA(Value: longint);',
  15670. ' class property pA: longint read fa write fa;',
  15671. ' class property pB: longint read geta write seta;',
  15672. ' end;',
  15673. ' TObjectClass = class of tobject;',
  15674. 'class function tobject.geta: longint; begin end;',
  15675. 'class procedure tobject.seta(value: longint); begin end;',
  15676. 'var',
  15677. ' b: boolean;',
  15678. ' Obj: tobject;',
  15679. ' Cla: tobjectclass;',
  15680. 'begin',
  15681. ' obj.pa:=obj.pa;',
  15682. ' obj.pb:=obj.pb;',
  15683. ' b:=obj.pa=4;',
  15684. ' b:=obj.pb=obj.pb;',
  15685. ' b:=5=obj.pa;',
  15686. ' cla.pa:=6;',
  15687. ' cla.pa:=cla.pa;',
  15688. ' cla.pb:=cla.pb;',
  15689. ' b:=cla.pa=7;',
  15690. ' b:=cla.pb=cla.pb;',
  15691. ' b:=8=cla.pa;',
  15692. ' tobject.pa:=9;',
  15693. ' tobject.pb:=tobject.pb;',
  15694. ' b:=tobject.pa=10;',
  15695. ' b:=11=tobject.pa;',
  15696. '']);
  15697. ConvertProgram;
  15698. CheckSource('TestClassOf_ClassProperty',
  15699. LinesToStr([ // statements
  15700. 'rtl.createClass(this, "TObject", null, function () {',
  15701. ' this.FA = 0;',
  15702. ' this.$init = function () {',
  15703. ' };',
  15704. ' this.$final = function () {',
  15705. ' };',
  15706. ' this.GetA = function () {',
  15707. ' var Result = 0;',
  15708. ' return Result;',
  15709. ' };',
  15710. ' this.SetA = function (Value) {',
  15711. ' };',
  15712. '});',
  15713. 'this.b = false;',
  15714. 'this.Obj = null;',
  15715. 'this.Cla = null;'
  15716. ]),
  15717. LinesToStr([ // $mod.$main
  15718. '$mod.TObject.FA = $mod.Obj.FA;',
  15719. '$mod.Obj.$class.SetA($mod.Obj.$class.GetA());',
  15720. '$mod.b = $mod.Obj.FA === 4;',
  15721. '$mod.b = $mod.Obj.$class.GetA() === $mod.Obj.$class.GetA();',
  15722. '$mod.b = 5 === $mod.Obj.FA;',
  15723. '$mod.TObject.FA = 6;',
  15724. '$mod.TObject.FA = $mod.Cla.FA;',
  15725. '$mod.Cla.SetA($mod.Cla.GetA());',
  15726. '$mod.b = $mod.Cla.FA === 7;',
  15727. '$mod.b = $mod.Cla.GetA() === $mod.Cla.GetA();',
  15728. '$mod.b = 8 === $mod.Cla.FA;',
  15729. '$mod.TObject.FA = 9;',
  15730. '$mod.TObject.SetA($mod.TObject.GetA());',
  15731. '$mod.b = $mod.TObject.FA === 10;',
  15732. '$mod.b = 11 === $mod.TObject.FA;',
  15733. '']));
  15734. end;
  15735. procedure TTestModule.TestClassOf_ClassMethodSelf;
  15736. begin
  15737. StartProgram(false);
  15738. Add('type');
  15739. Add(' TObject = class');
  15740. Add(' class var GlobalId: longint;');
  15741. Add(' class procedure ProcA;');
  15742. Add(' end;');
  15743. Add('class procedure tobject.proca;');
  15744. Add('var b: boolean;');
  15745. Add('begin');
  15746. Add(' b:=self=nil;');
  15747. Add(' b:=self.globalid=3;');
  15748. Add(' b:=4=self.globalid;');
  15749. Add(' self.globalid:=5;');
  15750. Add(' self.proca;');
  15751. Add('end;');
  15752. Add('begin');
  15753. ConvertProgram;
  15754. CheckSource('TestClassOf_ClassMethodSelf',
  15755. LinesToStr([ // statements
  15756. 'rtl.createClass(this, "TObject", null, function () {',
  15757. ' this.GlobalId = 0;',
  15758. ' this.$init = function () {',
  15759. ' };',
  15760. ' this.$final = function () {',
  15761. ' };',
  15762. ' this.ProcA = function () {',
  15763. ' var b = false;',
  15764. ' b = this === null;',
  15765. ' b = this.GlobalId === 3;',
  15766. ' b = 4 === this.GlobalId;',
  15767. ' $mod.TObject.GlobalId = 5;',
  15768. ' this.ProcA();',
  15769. ' };',
  15770. '});'
  15771. ]),
  15772. LinesToStr([ // $mod.$main
  15773. '']));
  15774. end;
  15775. procedure TTestModule.TestClassOf_TypeCast;
  15776. begin
  15777. StartProgram(false);
  15778. Add('type');
  15779. Add(' TObject = class');
  15780. Add(' class procedure {#TObject_DoIt}DoIt;');
  15781. Add(' end;');
  15782. Add(' TClass = class of TObject;');
  15783. Add(' TMobile = class');
  15784. Add(' class procedure {#TMobile_DoIt}DoIt;');
  15785. Add(' end;');
  15786. Add(' TMobileClass = class of TMobile;');
  15787. Add(' TCar = class(TMobile)');
  15788. Add(' class procedure {#TCar_DoIt}DoIt;');
  15789. Add(' end;');
  15790. Add(' TCarClass = class of TCar;');
  15791. Add('class procedure TObject.DoIt;');
  15792. Add('begin');
  15793. Add(' TClass(Self).{@TObject_DoIt}DoIt;');
  15794. Add(' TMobileClass(Self).{@TMobile_DoIt}DoIt;');
  15795. Add('end;');
  15796. Add('class procedure TMobile.DoIt;');
  15797. Add('begin');
  15798. Add(' TClass(Self).{@TObject_DoIt}DoIt;');
  15799. Add(' TMobileClass(Self).{@TMobile_DoIt}DoIt;');
  15800. Add(' TCarClass(Self).{@TCar_DoIt}DoIt;');
  15801. Add('end;');
  15802. Add('class procedure TCar.DoIt; begin end;');
  15803. Add('var');
  15804. Add(' ObjC: TClass;');
  15805. Add(' MobileC: TMobileClass;');
  15806. Add(' CarC: TCarClass;');
  15807. Add('begin');
  15808. Add(' ObjC.{@TObject_DoIt}DoIt;');
  15809. Add(' MobileC.{@TMobile_DoIt}DoIt;');
  15810. Add(' CarC.{@TCar_DoIt}DoIt;');
  15811. Add(' TClass(ObjC).{@TObject_DoIt}DoIt;');
  15812. Add(' TMobileClass(ObjC).{@TMobile_DoIt}DoIt;');
  15813. Add(' TCarClass(ObjC).{@TCar_DoIt}DoIt;');
  15814. Add(' TClass(MobileC).{@TObject_DoIt}DoIt;');
  15815. Add(' TMobileClass(MobileC).{@TMobile_DoIt}DoIt;');
  15816. Add(' TCarClass(MobileC).{@TCar_DoIt}DoIt;');
  15817. Add(' TClass(CarC).{@TObject_DoIt}DoIt;');
  15818. Add(' TMobileClass(CarC).{@TMobile_DoIt}DoIt;');
  15819. Add(' TCarClass(CarC).{@TCar_DoIt}DoIt;');
  15820. ConvertProgram;
  15821. CheckSource('TestClassOf_TypeCast',
  15822. LinesToStr([ // statements
  15823. 'rtl.createClass(this, "TObject", null, function () {',
  15824. ' this.$init = function () {',
  15825. ' };',
  15826. ' this.$final = function () {',
  15827. ' };',
  15828. ' this.DoIt = function () {',
  15829. ' this.DoIt();',
  15830. ' this.DoIt$1();',
  15831. ' };',
  15832. '});',
  15833. 'rtl.createClass(this, "TMobile", this.TObject, function () {',
  15834. ' this.DoIt$1 = function () {',
  15835. ' this.DoIt();',
  15836. ' this.DoIt$1();',
  15837. ' this.DoIt$2();',
  15838. ' };',
  15839. '});',
  15840. 'rtl.createClass(this, "TCar", this.TMobile, function () {',
  15841. ' this.DoIt$2 = function () {',
  15842. ' };',
  15843. '});',
  15844. 'this.ObjC = null;',
  15845. 'this.MobileC = null;',
  15846. 'this.CarC = null;',
  15847. '']),
  15848. LinesToStr([ // $mod.$main
  15849. '$mod.ObjC.DoIt();',
  15850. '$mod.MobileC.DoIt$1();',
  15851. '$mod.CarC.DoIt$2();',
  15852. '$mod.ObjC.DoIt();',
  15853. '$mod.ObjC.DoIt$1();',
  15854. '$mod.ObjC.DoIt$2();',
  15855. '$mod.MobileC.DoIt();',
  15856. '$mod.MobileC.DoIt$1();',
  15857. '$mod.MobileC.DoIt$2();',
  15858. '$mod.CarC.DoIt();',
  15859. '$mod.CarC.DoIt$1();',
  15860. '$mod.CarC.DoIt$2();',
  15861. '']));
  15862. end;
  15863. procedure TTestModule.TestClassOf_ImplicitFunctionCall;
  15864. begin
  15865. StartProgram(false);
  15866. Add('type');
  15867. Add(' TObject = class');
  15868. Add(' function CurNow: longint; ');
  15869. Add(' class function Now: longint; ');
  15870. Add(' end;');
  15871. Add('function TObject.CurNow: longint; begin end;');
  15872. Add('class function TObject.Now: longint; begin end;');
  15873. Add('var');
  15874. Add(' Obj: tobject;');
  15875. Add(' vI: longint;');
  15876. Add('begin');
  15877. Add(' obj.curnow;');
  15878. Add(' vi:=obj.curnow;');
  15879. Add(' tobject.now;');
  15880. Add(' vi:=tobject.now;');
  15881. ConvertProgram;
  15882. CheckSource('TestClassOf_ImplicitFunctionCall',
  15883. LinesToStr([ // statements
  15884. 'rtl.createClass(this, "TObject", null, function () {',
  15885. ' this.$init = function () {',
  15886. ' };',
  15887. ' this.$final = function () {',
  15888. ' };',
  15889. ' this.CurNow = function () {',
  15890. ' var Result = 0;',
  15891. ' return Result;',
  15892. ' };',
  15893. ' this.Now = function () {',
  15894. ' var Result = 0;',
  15895. ' return Result;',
  15896. ' };',
  15897. '});',
  15898. 'this.Obj = null;',
  15899. 'this.vI = 0;',
  15900. '']),
  15901. LinesToStr([ // $mod.$main
  15902. '$mod.Obj.CurNow();',
  15903. '$mod.vI = $mod.Obj.CurNow();',
  15904. '$mod.TObject.Now();',
  15905. '$mod.vI = $mod.TObject.Now();',
  15906. '']));
  15907. end;
  15908. procedure TTestModule.TestClassOf_Const;
  15909. begin
  15910. StartProgram(false);
  15911. Add([
  15912. 'type',
  15913. ' TObject = class',
  15914. ' end;',
  15915. ' TBird = TObject;',
  15916. ' TBirds = class of TBird;',
  15917. ' TEagles = TBirds;',
  15918. ' THawk = class(TBird);',
  15919. 'const',
  15920. ' Hawk: TEagles = THawk;',
  15921. ' DefaultBirdClasses : Array [1..2] of TEagles = (',
  15922. ' TBird,',
  15923. ' THawk',
  15924. ' );',
  15925. 'begin']);
  15926. ConvertProgram;
  15927. CheckSource('TestClassOf_Const',
  15928. LinesToStr([ // statements
  15929. 'rtl.createClass(this, "TObject", null, function () {',
  15930. ' this.$init = function () {',
  15931. ' };',
  15932. ' this.$final = function () {',
  15933. ' };',
  15934. '});',
  15935. 'rtl.createClass(this, "THawk", this.TObject, function () {',
  15936. '});',
  15937. 'this.Hawk = this.THawk;',
  15938. 'this.DefaultBirdClasses = [this.TObject, this.THawk];',
  15939. '']),
  15940. LinesToStr([ // $mod.$main
  15941. '']));
  15942. end;
  15943. procedure TTestModule.TestNestedClass_Alias;
  15944. begin
  15945. WithTypeInfo:=true;
  15946. StartProgram(false);
  15947. Add([
  15948. 'type',
  15949. ' TObject = class',
  15950. ' type TNested = type longint;',
  15951. ' end;',
  15952. 'type TAlias = type tobject.tnested;',
  15953. 'var i: tobject.tnested = 3;',
  15954. 'var j: TAlias = 4;',
  15955. 'begin',
  15956. ' if typeinfo(TAlias)=nil then ;',
  15957. ' if typeinfo(tobject.tnested)=nil then ;',
  15958. '']);
  15959. ConvertProgram;
  15960. CheckSource('TestNestedClass_Alias',
  15961. LinesToStr([ // statements
  15962. 'rtl.createClass(this, "TObject", null, function () {',
  15963. ' $mod.$rtti.$inherited("TObject.TNested", rtl.longint, {});',
  15964. ' this.$init = function () {',
  15965. ' };',
  15966. ' this.$final = function () {',
  15967. ' };',
  15968. '});',
  15969. 'this.$rtti.$inherited("TAlias", this.$rtti["TObject.TNested"], {});',
  15970. 'this.i = 3;',
  15971. 'this.j = 4;',
  15972. '']),
  15973. LinesToStr([ // $mod.$main
  15974. 'if ($mod.$rtti["TAlias"] === null) ;',
  15975. 'if ($mod.$rtti["TObject.TNested"] === null) ;',
  15976. '']));
  15977. end;
  15978. procedure TTestModule.TestNestedClass_Record;
  15979. begin
  15980. WithTypeInfo:=true;
  15981. StartProgram(false);
  15982. Add([
  15983. 'type',
  15984. ' TObject = class',
  15985. ' type TPoint = record',
  15986. ' x,y: byte;',
  15987. ' end;',
  15988. ' procedure DoIt(t: TPoint);',
  15989. ' end;',
  15990. 'procedure tobject.DoIt(t: TPoint);',
  15991. 'var p: TPoint;',
  15992. 'begin',
  15993. ' t.x:=t.y;',
  15994. ' p:=t;',
  15995. 'end;',
  15996. 'var',
  15997. ' p: tobject.tpoint = (x:2; y:4);',
  15998. ' o: TObject;',
  15999. 'begin',
  16000. ' p:=p;',
  16001. ' o.doit(p);',
  16002. '']);
  16003. ConvertProgram;
  16004. CheckSource('TestNestedClass_Record',
  16005. LinesToStr([ // statements
  16006. 'rtl.createClass(this, "TObject", null, function () {',
  16007. ' rtl.recNewT(this, "TPoint", function () {',
  16008. ' this.x = 0;',
  16009. ' this.y = 0;',
  16010. ' this.$eq = function (b) {',
  16011. ' return (this.x === b.x) && (this.y === b.y);',
  16012. ' };',
  16013. ' this.$assign = function (s) {',
  16014. ' this.x = s.x;',
  16015. ' this.y = s.y;',
  16016. ' return this;',
  16017. ' };',
  16018. ' var $r = $mod.$rtti.$Record("TObject.TPoint", {});',
  16019. ' $r.addField("x", rtl.byte);',
  16020. ' $r.addField("y", rtl.byte);',
  16021. ' });',
  16022. ' this.$init = function () {',
  16023. ' };',
  16024. ' this.$final = function () {',
  16025. ' };',
  16026. ' this.DoIt = function (t) {',
  16027. ' var p = this.TPoint.$new();',
  16028. ' t.x = t.y;',
  16029. ' p.$assign(t);',
  16030. ' };',
  16031. '});',
  16032. 'this.p = this.TObject.TPoint.$clone({',
  16033. ' x: 2,',
  16034. ' y: 4',
  16035. '});',
  16036. 'this.o = null;',
  16037. '']),
  16038. LinesToStr([ // $mod.$main
  16039. '$mod.p.$assign($mod.p);',
  16040. '$mod.o.DoIt($mod.TObject.TPoint.$clone($mod.p));',
  16041. '']));
  16042. end;
  16043. procedure TTestModule.TestNestedClass_Class;
  16044. begin
  16045. WithTypeInfo:=true;
  16046. StartProgram(false);
  16047. Add([
  16048. 'type',
  16049. ' TObject = class end;',
  16050. ' TBird = class',
  16051. ' type TLeg = class',
  16052. ' FId: longint;',
  16053. ' constructor Create;',
  16054. ' function Create(i: longint): TLeg;',
  16055. ' end;',
  16056. ' function DoIt(b: TBird): Tleg;',
  16057. ' end;',
  16058. 'constructor tbird.tleg.create;',
  16059. 'begin',
  16060. ' FId:=3;',
  16061. 'end;',
  16062. 'function tbird.tleg.Create(i: longint): TLeg;',
  16063. 'begin',
  16064. ' Create;',
  16065. ' Result:=TLeg.Create;',
  16066. ' Result:=TBird.TLeg.Create;',
  16067. ' Result:=Create(3);',
  16068. ' FId:=i;',
  16069. 'end;',
  16070. 'function tbird.DoIt(b: tbird): tleg;',
  16071. 'begin',
  16072. ' Result.Create;',
  16073. ' Result:=TLeg.Create;',
  16074. ' Result:=TBird.TLeg.Create;',
  16075. ' Result:=Result.Create(3);',
  16076. 'end;',
  16077. 'var',
  16078. ' b: Tbird.tleg;',
  16079. 'begin',
  16080. ' b.Create;',
  16081. ' b:=TBird.TLeg.Create;',
  16082. ' b:=b.Create(3);',
  16083. '']);
  16084. ConvertProgram;
  16085. CheckSource('TestNestedClass_Class',
  16086. LinesToStr([ // statements
  16087. 'rtl.createClass(this, "TObject", null, function () {',
  16088. ' this.$init = function () {',
  16089. ' };',
  16090. ' this.$final = function () {',
  16091. ' };',
  16092. '});',
  16093. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  16094. ' rtl.createClass(this, "TLeg", $mod.TObject, function () {',
  16095. ' this.$init = function () {',
  16096. ' $mod.TObject.$init.call(this);',
  16097. ' this.FId = 0;',
  16098. ' };',
  16099. ' this.Create = function () {',
  16100. ' this.FId = 3;',
  16101. ' return this;',
  16102. ' };',
  16103. ' this.Create$1 = function (i) {',
  16104. ' var Result = null;',
  16105. ' this.Create();',
  16106. ' Result = $mod.TBird.TLeg.$create("Create");',
  16107. ' Result = $mod.TBird.TLeg.$create("Create");',
  16108. ' Result = this.Create$1(3);',
  16109. ' this.FId = i;',
  16110. ' return Result;',
  16111. ' };',
  16112. ' }, "TBird.TLeg");',
  16113. ' this.DoIt = function (b) {',
  16114. ' var Result = null;',
  16115. ' Result.Create();',
  16116. ' Result = this.TLeg.$create("Create");',
  16117. ' Result = $mod.TBird.TLeg.$create("Create");',
  16118. ' Result = Result.Create$1(3);',
  16119. ' return Result;',
  16120. ' };',
  16121. '});',
  16122. 'this.b = null;',
  16123. '']),
  16124. LinesToStr([ // $mod.$main
  16125. '$mod.b.Create();',
  16126. '$mod.b = $mod.TBird.TLeg.$create("Create");',
  16127. '$mod.b = $mod.b.Create$1(3);',
  16128. '']));
  16129. end;
  16130. procedure TTestModule.TestExternalClass_Var;
  16131. begin
  16132. StartProgram(false);
  16133. Add([
  16134. '{$modeswitch externalclass}',
  16135. 'type',
  16136. ' TExtA = class external name ''ExtObj''',
  16137. ' Id: longint external name ''$Id'';',
  16138. ' B: longint;',
  16139. ' end;',
  16140. 'var Obj: TExtA;',
  16141. 'begin',
  16142. ' obj.id:=obj.id+1;',
  16143. ' obj.B:=obj.B+1;']);
  16144. ConvertProgram;
  16145. CheckSource('TestExternalClass_Var',
  16146. LinesToStr([ // statements
  16147. 'this.Obj = null;',
  16148. '']),
  16149. LinesToStr([ // $mod.$main
  16150. '$mod.Obj.$Id = $mod.Obj.$Id + 1;',
  16151. '$mod.Obj.B = $mod.Obj.B + 1;',
  16152. '']));
  16153. end;
  16154. procedure TTestModule.TestExternalClass_Const;
  16155. begin
  16156. StartProgram(false);
  16157. Add([
  16158. '{$modeswitch externalclass}',
  16159. 'type',
  16160. ' TExtA = class external name ''ExtObj''',
  16161. ' const Two: longint = 2;',
  16162. ' const Three = 3;',
  16163. ' const Id: longint;',
  16164. ' end;',
  16165. ' TExtB = class external name ''ExtB''',
  16166. ' A: TExtA;',
  16167. ' end;',
  16168. 'var',
  16169. ' A: texta;',
  16170. ' B: textb;',
  16171. ' i: longint;',
  16172. 'begin',
  16173. ' i:=a.two;',
  16174. ' i:=texta.two;',
  16175. ' i:=a.three;',
  16176. ' i:=texta.three;',
  16177. ' i:=a.id;',
  16178. ' i:=texta.id;',
  16179. '']);
  16180. ConvertProgram;
  16181. CheckSource('TestExternalClass_Const',
  16182. LinesToStr([ // statements
  16183. 'this.A = null;',
  16184. 'this.B = null;',
  16185. 'this.i = 0;',
  16186. '']),
  16187. LinesToStr([ // $mod.$main
  16188. '$mod.i = 2;',
  16189. '$mod.i = 2;',
  16190. '$mod.i = 3;',
  16191. '$mod.i = 3;',
  16192. '$mod.i = $mod.A.Id;',
  16193. '$mod.i = ExtObj.Id;',
  16194. '']));
  16195. end;
  16196. procedure TTestModule.TestExternalClass_Dollar;
  16197. begin
  16198. StartProgram(false);
  16199. Add([
  16200. '{$modeswitch externalclass}',
  16201. 'type',
  16202. ' TExtA = class external name ''$''',
  16203. ' Id: longint external name ''$'';',
  16204. ' function Bla(i: longint): longint; external name ''$'';',
  16205. ' end;',
  16206. 'function dollar(k: longint): longint; external name ''$'';',
  16207. 'var Obj: TExtA;',
  16208. 'begin',
  16209. ' dollar(1);',
  16210. ' obj.id:=obj.id+2;',
  16211. ' obj.Bla(3);',
  16212. '']);
  16213. ConvertProgram;
  16214. CheckSource('TestExternalClass_Dollar',
  16215. LinesToStr([ // statements
  16216. 'this.Obj = null;',
  16217. '']),
  16218. LinesToStr([ // $mod.$main
  16219. '$(1);',
  16220. '$mod.Obj.$ = $mod.Obj.$ + 2;',
  16221. '$mod.Obj.$(3);',
  16222. '']));
  16223. end;
  16224. procedure TTestModule.TestExternalClass_DuplicateVarFail;
  16225. begin
  16226. StartProgram(false);
  16227. Add('{$modeswitch externalclass}');
  16228. Add('type');
  16229. Add(' TExtA = class external name ''ExtA''');
  16230. Add(' Id: longint external name ''$Id'';');
  16231. Add(' end;');
  16232. Add(' TExtB = class external ''lib'' name ''ExtB''(TExtA)');
  16233. Add(' Id: longint;');
  16234. Add(' end;');
  16235. Add('begin');
  16236. SetExpectedPasResolverError('Duplicate identifier "Id" at test1.pp(6,5)',nDuplicateIdentifier);
  16237. ConvertProgram;
  16238. end;
  16239. procedure TTestModule.TestExternalClass_Method;
  16240. begin
  16241. StartProgram(false);
  16242. Add(['{$modeswitch externalclass}',
  16243. 'type',
  16244. ' TExtA = class external name ''ExtObj''',
  16245. ' procedure DoIt(Id: longint = 1); external name ''$Execute'';',
  16246. ' procedure DoSome(Id: longint = 1);',
  16247. ' end;',
  16248. 'var Obj: texta;',
  16249. 'begin',
  16250. ' obj.doit;',
  16251. ' obj.doit();',
  16252. ' obj.doit(2);',
  16253. ' with obj do begin',
  16254. ' doit;',
  16255. ' doit();',
  16256. ' doit(3);',
  16257. ' end;']);
  16258. ConvertProgram;
  16259. CheckSource('TestExternalClass_Method',
  16260. LinesToStr([ // statements
  16261. 'this.Obj = null;',
  16262. '']),
  16263. LinesToStr([ // $mod.$main
  16264. '$mod.Obj.$Execute(1);',
  16265. '$mod.Obj.$Execute(1);',
  16266. '$mod.Obj.$Execute(2);',
  16267. 'var $with = $mod.Obj;',
  16268. '$with.$Execute(1);',
  16269. '$with.$Execute(1);',
  16270. '$with.$Execute(3);',
  16271. '']));
  16272. end;
  16273. procedure TTestModule.TestExternalClass_ClassMethod;
  16274. begin
  16275. StartProgram(false);
  16276. Add([
  16277. '{$modeswitch externalclass}',
  16278. 'type',
  16279. ' TExtA = class external name ''ExtObj''',
  16280. ' class procedure DoIt(Id: longint = 1); external name ''$Execute'';',
  16281. ' end;',
  16282. ' TExtB = TExtA;',
  16283. 'var p: Pointer;',
  16284. 'begin',
  16285. ' texta.doit;',
  16286. ' texta.doit();',
  16287. ' texta.doit(2);',
  16288. ' p:[email protected];',
  16289. ' with texta do begin',
  16290. ' doit;',
  16291. ' doit();',
  16292. ' doit(3);',
  16293. ' p:=@DoIt;',
  16294. ' end;',
  16295. ' textb.doit;',
  16296. ' textb.doit();',
  16297. ' textb.doit(4);',
  16298. ' with textb do begin',
  16299. ' doit;',
  16300. ' doit();',
  16301. ' doit(5);',
  16302. ' end;',
  16303. '']);
  16304. ConvertProgram;
  16305. CheckSource('TestExternalClass_ClassMethod',
  16306. LinesToStr([ // statements
  16307. 'this.p = null;',
  16308. '']),
  16309. LinesToStr([ // $mod.$main
  16310. 'ExtObj.$Execute(1);',
  16311. 'ExtObj.$Execute(1);',
  16312. 'ExtObj.$Execute(2);',
  16313. '$mod.p = rtl.createCallback(ExtObj, "$Execute");',
  16314. 'ExtObj.$Execute(1);',
  16315. 'ExtObj.$Execute(1);',
  16316. 'ExtObj.$Execute(3);',
  16317. '$mod.p = rtl.createCallback(ExtObj, "$Execute");',
  16318. 'ExtObj.$Execute(1);',
  16319. 'ExtObj.$Execute(1);',
  16320. 'ExtObj.$Execute(4);',
  16321. 'ExtObj.$Execute(1);',
  16322. 'ExtObj.$Execute(1);',
  16323. 'ExtObj.$Execute(5);',
  16324. '']));
  16325. end;
  16326. procedure TTestModule.TestExternalClass_ClassMethodStatic;
  16327. begin
  16328. StartProgram(false);
  16329. Add([
  16330. '{$modeswitch externalclass}',
  16331. 'type',
  16332. ' TExtA = class external name ''ExtObj''',
  16333. ' class procedure DoIt(Id: longint = 1); static;',
  16334. ' end;',
  16335. 'var p: Pointer;',
  16336. 'begin',
  16337. ' texta.doit;',
  16338. ' texta.doit();',
  16339. ' texta.doit(2);',
  16340. ' p:[email protected];',
  16341. ' with texta do begin',
  16342. ' doit;',
  16343. ' doit();',
  16344. ' doit(3);',
  16345. ' p:=@DoIt;',
  16346. ' end;',
  16347. '']);
  16348. ConvertProgram;
  16349. CheckSource('TestExternalClass_ClassMethodStatic',
  16350. LinesToStr([ // statements
  16351. 'this.p = null;',
  16352. '']),
  16353. LinesToStr([ // $mod.$main
  16354. 'ExtObj.DoIt(1);',
  16355. 'ExtObj.DoIt(1);',
  16356. 'ExtObj.DoIt(2);',
  16357. '$mod.p = ExtObj.DoIt;',
  16358. 'ExtObj.DoIt(1);',
  16359. 'ExtObj.DoIt(1);',
  16360. 'ExtObj.DoIt(3);',
  16361. '$mod.p = ExtObj.DoIt;',
  16362. '']));
  16363. end;
  16364. procedure TTestModule.TestExternalClass_FunctionResultInTypeCast;
  16365. begin
  16366. StartProgram(false);
  16367. Add([
  16368. '{$modeswitch externalclass}',
  16369. 'type',
  16370. ' TBird = class external name ''Array''',
  16371. ' end;',
  16372. 'function GetPtr: Pointer;',
  16373. 'begin',
  16374. 'end;',
  16375. 'procedure Write(const p);',
  16376. 'begin',
  16377. 'end;',
  16378. 'procedure WriteLn; varargs;',
  16379. 'begin',
  16380. 'end;',
  16381. 'begin',
  16382. ' if TBird(GetPtr)=nil then ;',
  16383. ' Write(GetPtr);',
  16384. ' WriteLn(GetPtr);',
  16385. ' Write(TBird(GetPtr));',
  16386. ' WriteLn(TBird(GetPtr));',
  16387. '']);
  16388. ConvertProgram;
  16389. CheckSource('TestFunctionResultInTypeCast',
  16390. LinesToStr([ // statements
  16391. 'this.GetPtr = function () {',
  16392. ' var Result = null;',
  16393. ' return Result;',
  16394. '};',
  16395. 'this.Write = function (p) {',
  16396. '};',
  16397. 'this.WriteLn = function () {',
  16398. '};',
  16399. '']),
  16400. LinesToStr([
  16401. 'if ($mod.GetPtr() === null) ;',
  16402. '$mod.Write($mod.GetPtr());',
  16403. '$mod.WriteLn($mod.GetPtr());',
  16404. '$mod.Write($mod.GetPtr());',
  16405. '$mod.WriteLn($mod.GetPtr());',
  16406. '']));
  16407. end;
  16408. procedure TTestModule.TestExternalClass_NonExternalOverride;
  16409. begin
  16410. StartProgram(false);
  16411. Add([
  16412. '{$modeswitch externalclass}',
  16413. 'type',
  16414. ' TExtA = class external name ''ExtObjA''',
  16415. ' procedure ProcA; virtual;',
  16416. ' procedure ProcB; virtual;',
  16417. ' end;',
  16418. ' TExtB = class external name ''ExtObjB'' (TExtA)',
  16419. ' end;',
  16420. ' TExtC = class (TExtB)',
  16421. ' procedure ProcA; override;',
  16422. ' end;',
  16423. 'procedure TExtC.ProcA;',
  16424. 'begin',
  16425. ' ProcA;',
  16426. ' Self.ProcA;',
  16427. ' ProcB;',
  16428. ' Self.ProcB;',
  16429. 'end;',
  16430. 'var',
  16431. ' A: texta;',
  16432. ' B: textb;',
  16433. ' C: textc;',
  16434. 'begin',
  16435. ' a.proca;',
  16436. ' b.proca;',
  16437. ' c.proca;']);
  16438. ConvertProgram;
  16439. CheckSource('TestExternalClass_NonExternalOverride',
  16440. LinesToStr([ // statements
  16441. 'rtl.createClassExt(this, "TExtC", ExtObjB, "", function () {',
  16442. ' this.$init = function () {',
  16443. ' };',
  16444. ' this.$final = function () {',
  16445. ' };',
  16446. ' this.ProcA = function () {',
  16447. ' this.ProcA();',
  16448. ' this.ProcA();',
  16449. ' this.ProcB();',
  16450. ' this.ProcB();',
  16451. ' };',
  16452. '});',
  16453. 'this.A = null;',
  16454. 'this.B = null;',
  16455. 'this.C = null;',
  16456. '']),
  16457. LinesToStr([ // $mod.$main
  16458. '$mod.A.ProcA();',
  16459. '$mod.B.ProcA();',
  16460. '$mod.C.ProcA();',
  16461. '']));
  16462. end;
  16463. procedure TTestModule.TestExternalClass_OverloadHint;
  16464. begin
  16465. StartProgram(false);
  16466. Add([
  16467. '{$modeswitch externalclass}',
  16468. 'type',
  16469. ' TExtA = class external name ''ExtObjA''',
  16470. ' procedure DoIt;',
  16471. ' procedure DoIt(i: longint);',
  16472. ' end;',
  16473. 'begin',
  16474. '']);
  16475. ConvertProgram;
  16476. CheckResolverUnexpectedHints(true);
  16477. CheckSource('TestExternalClass_OverloadHint',
  16478. LinesToStr([ // statements
  16479. '']),
  16480. LinesToStr([ // $mod.$main
  16481. '']));
  16482. end;
  16483. procedure TTestModule.TestExternalClass_SameNamePublishedProperty;
  16484. begin
  16485. StartProgram(false);
  16486. Add([
  16487. '{$modeswitch externalclass}',
  16488. 'type',
  16489. ' JSwiper = class external name ''Swiper''',
  16490. ' constructor New;',
  16491. ' end;',
  16492. ' TObject = class',
  16493. ' private',
  16494. ' FSwiper: JSwiper;',
  16495. ' published',
  16496. ' property Swiper: JSwiper read FSwiper write FSwiper;',
  16497. ' end;',
  16498. 'begin',
  16499. ' JSwiper.new;',
  16500. '']);
  16501. ConvertProgram;
  16502. CheckSource('TestExternalClass_SameNamePublishedProperty',
  16503. LinesToStr([ // statements
  16504. 'rtl.createClass(this, "TObject", null, function () {',
  16505. ' this.$init = function () {',
  16506. ' this.FSwiper = null;',
  16507. ' };',
  16508. ' this.$final = function () {',
  16509. ' this.FSwiper = undefined;',
  16510. ' };',
  16511. ' var $r = this.$rtti;',
  16512. ' $r.addProperty("Swiper", 0, $mod.$rtti["JSwiper"], "FSwiper", "FSwiper");',
  16513. '});',
  16514. '']),
  16515. LinesToStr([ // $mod.$main
  16516. 'new Swiper();',
  16517. '']));
  16518. end;
  16519. procedure TTestModule.TestExternalClass_Property;
  16520. begin
  16521. StartProgram(false);
  16522. Add([
  16523. '{$modeswitch externalclass}',
  16524. 'type',
  16525. ' TExtA = class external name ''ExtA''',
  16526. ' function getYear: longint;',
  16527. ' procedure setYear(Value: longint);',
  16528. ' property Year: longint read getyear write setyear;',
  16529. ' end;',
  16530. ' TExtB = class (TExtA)',
  16531. ' procedure OtherSetYear(Value: longint);',
  16532. ' property year write othersetyear;',
  16533. ' end;',
  16534. 'procedure textb.othersetyear(value: longint);',
  16535. 'begin',
  16536. ' setYear(Value+4);',
  16537. 'end;',
  16538. 'var',
  16539. ' A: texta;',
  16540. ' B: textb;',
  16541. 'begin',
  16542. ' a.year:=a.year+1;',
  16543. ' b.year:=b.year+2;']);
  16544. ConvertProgram;
  16545. CheckSource('TestExternalClass_NonExternalOverride',
  16546. LinesToStr([ // statements
  16547. 'rtl.createClassExt(this, "TExtB", ExtA, "", function () {',
  16548. ' this.$init = function () {',
  16549. ' };',
  16550. ' this.$final = function () {',
  16551. ' };',
  16552. ' this.OtherSetYear = function (Value) {',
  16553. ' this.setYear(Value+4);',
  16554. ' };',
  16555. '});',
  16556. 'this.A = null;',
  16557. 'this.B = null;',
  16558. '']),
  16559. LinesToStr([ // $mod.$main
  16560. '$mod.A.setYear($mod.A.getYear()+1);',
  16561. '$mod.B.OtherSetYear($mod.B.getYear()+2);',
  16562. '']));
  16563. end;
  16564. procedure TTestModule.TestExternalClass_PropertyDate;
  16565. begin
  16566. StartProgram(false);
  16567. Add([
  16568. '{$modeswitch externalclass}',
  16569. 'type',
  16570. ' TExtA = class external name ''ExtA''',
  16571. ' end;',
  16572. ' TExtB = class (TExtA)',
  16573. ' FDate: string;',
  16574. ' property Date: string read FDate write FDate;',
  16575. ' property ExtA: string read FDate write FDate;',
  16576. ' end;',
  16577. ' {$M+}',
  16578. ' TObject = class',
  16579. ' FDate: string;',
  16580. ' published',
  16581. ' property Date: string read FDate write FDate;',
  16582. ' property ExtA: string read FDate write FDate;',
  16583. ' end;',
  16584. 'var',
  16585. ' B: textb;',
  16586. ' o: TObject;',
  16587. 'begin',
  16588. ' b.date:=b.exta;',
  16589. ' o.date:=o.exta;']);
  16590. ConvertProgram;
  16591. CheckSource('TestExternalClass_PropertyDate',
  16592. LinesToStr([ // statements
  16593. 'rtl.createClassExt(this, "TExtB", ExtA, "", function () {',
  16594. ' this.$init = function () {',
  16595. ' this.FDate = "";',
  16596. ' };',
  16597. ' this.$final = function () {',
  16598. ' };',
  16599. '});',
  16600. 'rtl.createClass(this, "TObject", null, function () {',
  16601. ' this.$init = function () {',
  16602. ' this.FDate = "";',
  16603. ' };',
  16604. ' this.$final = function () {',
  16605. ' };',
  16606. ' var $r = this.$rtti;',
  16607. ' $r.addField("FDate", rtl.string);',
  16608. ' $r.addProperty("Date", 0, rtl.string, "FDate", "FDate");',
  16609. ' $r.addProperty("ExtA", 0, rtl.string, "FDate", "FDate");',
  16610. '});',
  16611. 'this.B = null;',
  16612. 'this.o = null;',
  16613. '']),
  16614. LinesToStr([ // $mod.$main
  16615. '$mod.B.FDate = $mod.B.FDate;',
  16616. '$mod.o.FDate = $mod.o.FDate;',
  16617. '']));
  16618. end;
  16619. procedure TTestModule.TestExternalClass_ClassProperty;
  16620. begin
  16621. StartProgram(false);
  16622. Add('{$modeswitch externalclass}');
  16623. Add('type');
  16624. Add(' TExtA = class external name ''ExtA''');
  16625. Add(' class function getYear: longint;');
  16626. Add(' class procedure setYear(Value: longint);');
  16627. Add(' class property Year: longint read getyear write setyear;');
  16628. Add(' end;');
  16629. Add(' TExtB = class (TExtA)');
  16630. Add(' class function GetCentury: longint;');
  16631. Add(' class procedure SetCentury(Value: longint);');
  16632. Add(' class property Century: longint read getcentury write setcentury;');
  16633. Add(' end;');
  16634. Add('class function textb.getcentury: longint;');
  16635. Add('begin');
  16636. Add('end;');
  16637. Add('class procedure textb.setcentury(value: longint);');
  16638. Add('begin');
  16639. Add(' setyear(value+11);');
  16640. Add(' texta.year:=texta.year+12;');
  16641. Add(' year:=year+13;');
  16642. Add(' textb.century:=textb.century+14;');
  16643. Add(' century:=century+15;');
  16644. Add('end;');
  16645. Add('var');
  16646. Add(' A: texta;');
  16647. Add(' B: textb;');
  16648. Add('begin');
  16649. Add(' texta.year:=texta.year+1;');
  16650. Add(' textb.year:=textb.year+2;');
  16651. Add(' TextA.year:=TextA.year+3;');
  16652. Add(' b.year:=b.year+4;');
  16653. Add(' textb.century:=textb.century+5;');
  16654. Add(' b.century:=b.century+6;');
  16655. ConvertProgram;
  16656. CheckSource('TestExternalClass_ClassProperty',
  16657. LinesToStr([ // statements
  16658. 'rtl.createClassExt(this, "TExtB", ExtA, "", function () {',
  16659. ' this.$init = function () {',
  16660. ' };',
  16661. ' this.$final = function () {',
  16662. ' };',
  16663. ' this.GetCentury = function () {',
  16664. ' var Result = 0;',
  16665. ' return Result;',
  16666. ' };',
  16667. ' this.SetCentury = function (Value) {',
  16668. ' this.setYear(Value + 11);',
  16669. ' ExtA.setYear(ExtA.getYear() + 12);',
  16670. ' this.setYear(this.getYear() + 13);',
  16671. ' $mod.TExtB.SetCentury($mod.TExtB.GetCentury() + 14);',
  16672. ' this.SetCentury(this.GetCentury() + 15);',
  16673. ' };',
  16674. '});',
  16675. 'this.A = null;',
  16676. 'this.B = null;',
  16677. '']),
  16678. LinesToStr([ // $mod.$main
  16679. 'ExtA.setYear(ExtA.getYear() + 1);',
  16680. '$mod.TExtB.setYear($mod.TExtB.getYear() + 2);',
  16681. 'ExtA.setYear(ExtA.getYear() + 3);',
  16682. '$mod.B.setYear($mod.B.getYear() + 4);',
  16683. '$mod.TExtB.SetCentury($mod.TExtB.GetCentury() + 5);',
  16684. '$mod.B.$class.SetCentury($mod.B.$class.GetCentury() + 6);',
  16685. '']));
  16686. end;
  16687. procedure TTestModule.TestExternalClass_ClassOf;
  16688. begin
  16689. StartProgram(false);
  16690. Add('{$modeswitch externalclass}');
  16691. Add('type');
  16692. Add(' TExtA = class external name ''ExtA''');
  16693. Add(' procedure ProcA; virtual;');
  16694. Add(' procedure ProcB; virtual;');
  16695. Add(' end;');
  16696. Add(' TExtAClass = class of TExtA;');
  16697. Add(' TExtB = class external name ''ExtB'' (TExtA)');
  16698. Add(' end;');
  16699. Add(' TExtBClass = class of TExtB;');
  16700. Add(' TExtC = class (TExtB)');
  16701. Add(' procedure ProcA; override;');
  16702. Add(' end;');
  16703. Add(' TExtCClass = class of TExtC;');
  16704. Add('procedure TExtC.ProcA; begin end;');
  16705. Add('var');
  16706. Add(' A: texta; ClA: TExtAClass;');
  16707. Add(' B: textb; ClB: TExtBClass;');
  16708. Add(' C: textc; ClC: TExtCClass;');
  16709. Add('begin');
  16710. Add(' ClA:=texta;');
  16711. Add(' ClA:=textb;');
  16712. Add(' ClA:=textc;');
  16713. Add(' ClB:=textb;');
  16714. Add(' ClB:=textc;');
  16715. Add(' ClC:=textc;');
  16716. ConvertProgram;
  16717. CheckSource('TestExternalClass_ClassOf',
  16718. LinesToStr([ // statements
  16719. 'rtl.createClassExt(this, "TExtC", ExtB, "", function () {',
  16720. ' this.$init = function () {',
  16721. ' };',
  16722. ' this.$final = function () {',
  16723. ' };',
  16724. ' this.ProcA = function () {',
  16725. ' };',
  16726. '});',
  16727. 'this.A = null;',
  16728. 'this.ClA = null;',
  16729. 'this.B = null;',
  16730. 'this.ClB = null;',
  16731. 'this.C = null;',
  16732. 'this.ClC = null;',
  16733. '']),
  16734. LinesToStr([ // $mod.$main
  16735. '$mod.ClA = ExtA;',
  16736. '$mod.ClA = ExtB;',
  16737. '$mod.ClA = $mod.TExtC;',
  16738. '$mod.ClB = ExtB;',
  16739. '$mod.ClB = $mod.TExtC;',
  16740. '$mod.ClC = $mod.TExtC;',
  16741. '']));
  16742. end;
  16743. procedure TTestModule.TestExternalClass_ClassOtherUnit;
  16744. begin
  16745. AddModuleWithIntfImplSrc('unit2.pas',
  16746. LinesToStr([
  16747. '{$modeswitch externalclass}',
  16748. 'type',
  16749. ' TExtA = class external name ''ExtA''',
  16750. ' class var Id: longint;',
  16751. ' end;',
  16752. '']),
  16753. '');
  16754. StartUnit(true);
  16755. Add('interface');
  16756. Add('uses unit2;');
  16757. Add('implementation');
  16758. Add('begin');
  16759. Add(' unit2.texta.id:=unit2.texta.id+1;');
  16760. ConvertUnit;
  16761. CheckSource('TestExternalClass_ClassOtherUnit',
  16762. LinesToStr([
  16763. '']),
  16764. LinesToStr([
  16765. 'ExtA.Id = ExtA.Id + 1;',
  16766. '']));
  16767. end;
  16768. procedure TTestModule.TestExternalClass_Is;
  16769. begin
  16770. StartProgram(false);
  16771. Add([
  16772. '{$modeswitch externalclass}',
  16773. 'type',
  16774. ' TExtA = class external name ''ExtA''',
  16775. ' end;',
  16776. ' TExtAClass = class of TExtA;',
  16777. ' TExtB = class external name ''ExtB'' (TExtA)',
  16778. ' end;',
  16779. ' TExtBClass = class of TExtB;',
  16780. ' TExtC = class (TExtB)',
  16781. ' end;',
  16782. ' TExtCClass = class of TExtC;',
  16783. 'var',
  16784. ' A: texta; ClA: TExtAClass;',
  16785. ' B: textb; ClB: TExtBClass;',
  16786. ' C: textc; ClC: TExtCClass;',
  16787. 'begin',
  16788. ' if a is textb then ;',
  16789. ' if a is textc then ;',
  16790. ' if b is textc then ;',
  16791. ' if cla is textb then ;',
  16792. ' if cla is textc then ;',
  16793. ' if clb is textc then ;',
  16794. ' try',
  16795. ' except',
  16796. ' on TExtA do ;',
  16797. ' on e: TExtB do ;',
  16798. ' end;',
  16799. '']);
  16800. ConvertProgram;
  16801. CheckSource('TestExternalClass_Is',
  16802. LinesToStr([ // statements
  16803. 'rtl.createClassExt(this, "TExtC", ExtB, "", function () {',
  16804. ' this.$init = function () {',
  16805. ' };',
  16806. ' this.$final = function () {',
  16807. ' };',
  16808. '});',
  16809. 'this.A = null;',
  16810. 'this.ClA = null;',
  16811. 'this.B = null;',
  16812. 'this.ClB = null;',
  16813. 'this.C = null;',
  16814. 'this.ClC = null;',
  16815. '']),
  16816. LinesToStr([ // $mod.$main
  16817. 'if (rtl.isExt($mod.A, ExtB)) ;',
  16818. 'if ($mod.TExtC.isPrototypeOf($mod.A)) ;',
  16819. 'if ($mod.TExtC.isPrototypeOf($mod.B)) ;',
  16820. 'if (rtl.isExt($mod.ClA, ExtB)) ;',
  16821. 'if (rtl.is($mod.ClA, $mod.TExtC)) ;',
  16822. 'if (rtl.is($mod.ClB, $mod.TExtC)) ;',
  16823. 'try {} catch ($e) {',
  16824. ' if (rtl.isExt($e,ExtA)) {}',
  16825. ' else if (rtl.isExt($e,ExtB)) {',
  16826. ' var e = $e;',
  16827. ' } else throw $e',
  16828. '};',
  16829. '']));
  16830. end;
  16831. procedure TTestModule.TestExternalClass_As;
  16832. begin
  16833. StartProgram(false);
  16834. Add('{$modeswitch externalclass}');
  16835. Add('type');
  16836. Add(' TExtA = class external name ''ExtA''');
  16837. Add(' end;');
  16838. Add(' TExtB = class external name ''ExtB'' (TExtA)');
  16839. Add(' end;');
  16840. Add(' TExtC = class (TExtB)');
  16841. Add(' end;');
  16842. Add('var');
  16843. Add(' A: texta;');
  16844. Add(' B: textb;');
  16845. Add(' C: textc;');
  16846. Add('begin');
  16847. Add(' b:=a as textb;');
  16848. Add(' c:=a as textc;');
  16849. Add(' c:=b as textc;');
  16850. ConvertProgram;
  16851. CheckSource('TestExternalClass_Is',
  16852. LinesToStr([ // statements
  16853. 'rtl.createClassExt(this, "TExtC", ExtB, "", function () {',
  16854. ' this.$init = function () {',
  16855. ' };',
  16856. ' this.$final = function () {',
  16857. ' };',
  16858. '});',
  16859. 'this.A = null;',
  16860. 'this.B = null;',
  16861. 'this.C = null;',
  16862. '']),
  16863. LinesToStr([ // $mod.$main
  16864. '$mod.B = rtl.asExt($mod.A, ExtB);',
  16865. '$mod.C = rtl.as($mod.A, $mod.TExtC);',
  16866. '$mod.C = rtl.as($mod.B, $mod.TExtC);',
  16867. '']));
  16868. end;
  16869. procedure TTestModule.TestExternalClass_DestructorFail;
  16870. begin
  16871. StartProgram(false);
  16872. Add('{$modeswitch externalclass}');
  16873. Add('type');
  16874. Add(' TExtA = class external name ''ExtA''');
  16875. Add(' destructor Free;');
  16876. Add(' end;');
  16877. SetExpectedPasResolverError('Pascal element not supported: destructor',
  16878. nPasElementNotSupported);
  16879. ConvertProgram;
  16880. end;
  16881. procedure TTestModule.TestExternalClass_New;
  16882. begin
  16883. StartProgram(false);
  16884. Add([
  16885. '{$modeswitch externalclass}',
  16886. 'type',
  16887. ' TExtA = class external name ''ExtA''',
  16888. ' constructor New;',
  16889. ' constructor New(i: longint; j: longint = 2);',
  16890. ' end;',
  16891. 'var',
  16892. ' A: texta;',
  16893. 'begin',
  16894. ' a:=texta.new;',
  16895. ' a:=texta(texta.new);',
  16896. ' a:=texta.new();',
  16897. ' a:=texta.new(1);',
  16898. ' with texta do begin',
  16899. ' a:=new;',
  16900. ' a:=new();',
  16901. ' a:=new(2);',
  16902. ' end;',
  16903. ' a:=test1.texta.new;',
  16904. ' a:=test1.texta.new();',
  16905. ' a:=test1.texta.new(3);',
  16906. '']);
  16907. ConvertProgram;
  16908. CheckSource('TestExternalClass_New',
  16909. LinesToStr([ // statements
  16910. 'this.A = null;',
  16911. '']),
  16912. LinesToStr([ // $mod.$main
  16913. '$mod.A = new ExtA();',
  16914. '$mod.A = new ExtA();',
  16915. '$mod.A = new ExtA();',
  16916. '$mod.A = new ExtA(1,2);',
  16917. '$mod.A = new ExtA();',
  16918. '$mod.A = new ExtA();',
  16919. '$mod.A = new ExtA(2,2);',
  16920. '$mod.A = new ExtA();',
  16921. '$mod.A = new ExtA();',
  16922. '$mod.A = new ExtA(3,2);',
  16923. '']));
  16924. end;
  16925. procedure TTestModule.TestExternalClass_ClassOf_New;
  16926. begin
  16927. StartProgram(false);
  16928. Add('{$modeswitch externalclass}');
  16929. Add('type');
  16930. Add(' TExtAClass = class of TExtA;');
  16931. Add(' TExtA = class external name ''ExtA''');
  16932. Add(' C: TExtAClass;');
  16933. Add(' constructor New;');
  16934. Add(' end;');
  16935. Add('var');
  16936. Add(' A: texta;');
  16937. Add(' C: textaclass;');
  16938. Add('begin');
  16939. Add(' a:=c.new;');
  16940. Add(' a:=c.new();');
  16941. Add(' with C do begin');
  16942. Add(' a:=new;');
  16943. Add(' a:=new();');
  16944. Add(' end;');
  16945. Add(' a:=test1.c.new;');
  16946. Add(' a:=test1.c.new();');
  16947. Add(' a:=A.c.new();');
  16948. ConvertProgram;
  16949. CheckSource('TestExternalClass_ClassOf_New',
  16950. LinesToStr([ // statements
  16951. 'this.A = null;',
  16952. 'this.C = null;',
  16953. '']),
  16954. LinesToStr([ // $mod.$main
  16955. '$mod.A = new $mod.C();',
  16956. '$mod.A = new $mod.C();',
  16957. 'var $with = $mod.C;',
  16958. '$mod.A = new $with();',
  16959. '$mod.A = new $with();',
  16960. '$mod.A = new $mod.C();',
  16961. '$mod.A = new $mod.C();',
  16962. '$mod.A = new $mod.A.C();',
  16963. '']));
  16964. end;
  16965. procedure TTestModule.TestExternalClass_FuncClassOf_New;
  16966. begin
  16967. StartProgram(false);
  16968. Add([
  16969. '{$modeswitch externalclass}',
  16970. 'type',
  16971. ' TExtAClass = class of TExtA;',
  16972. ' TExtA = class external name ''ExtA''',
  16973. ' constructor New;',
  16974. ' end;',
  16975. 'function GetCreator: TExtAClass;',
  16976. 'begin',
  16977. ' Result:=TExtA;',
  16978. 'end;',
  16979. 'var',
  16980. ' A: texta;',
  16981. 'begin',
  16982. ' a:=getcreator.new;',
  16983. ' a:=getcreator().new;',
  16984. ' a:=getcreator().new();',
  16985. ' a:=getcreator.new();',
  16986. ' with getcreator do begin',
  16987. ' a:=new;',
  16988. ' a:=new();',
  16989. ' end;']);
  16990. ConvertProgram;
  16991. CheckSource('TestExternalClass_FuncClassOf_New',
  16992. LinesToStr([ // statements
  16993. 'this.GetCreator = function () {',
  16994. ' var Result = null;',
  16995. ' Result = ExtA;',
  16996. ' return Result;',
  16997. '};',
  16998. 'this.A = null;',
  16999. '']),
  17000. LinesToStr([ // $mod.$main
  17001. '$mod.A = new ($mod.GetCreator())();',
  17002. '$mod.A = new ($mod.GetCreator())();',
  17003. '$mod.A = new ($mod.GetCreator())();',
  17004. '$mod.A = new ($mod.GetCreator())();',
  17005. 'var $with = $mod.GetCreator();',
  17006. '$mod.A = new $with();',
  17007. '$mod.A = new $with();',
  17008. '']));
  17009. end;
  17010. procedure TTestModule.TestExternalClass_New_PasClassFail;
  17011. begin
  17012. StartProgram(false);
  17013. Add([
  17014. '{$modeswitch externalclass}',
  17015. 'type',
  17016. ' TExtA = class external name ''ExtA''',
  17017. ' constructor New;',
  17018. ' end;',
  17019. ' TBird = class(TExtA)',
  17020. ' end;',
  17021. 'begin',
  17022. ' TBird.new;',
  17023. '']);
  17024. SetExpectedPasResolverError(sJSNewNotSupported,nJSNewNotSupported);
  17025. ConvertProgram;
  17026. end;
  17027. procedure TTestModule.TestExternalClass_New_PasClassBracketsFail;
  17028. begin
  17029. StartProgram(false);
  17030. Add([
  17031. '{$modeswitch externalclass}',
  17032. 'type',
  17033. ' TExtA = class external name ''ExtA''',
  17034. ' constructor New;',
  17035. ' end;',
  17036. ' TBird = class(TExtA)',
  17037. ' end;',
  17038. 'begin',
  17039. ' TBird.new();',
  17040. '']);
  17041. SetExpectedPasResolverError(sJSNewNotSupported,nJSNewNotSupported);
  17042. ConvertProgram;
  17043. end;
  17044. procedure TTestModule.TestExternalClass_NewExtName;
  17045. begin
  17046. StartProgram(false);
  17047. Add([
  17048. '{$modeswitch externalclass}',
  17049. 'type',
  17050. ' TExtA = class external name ''ExtA''',
  17051. ' constructor New; external name ''Other'';',
  17052. ' constructor New(i: longint; j: longint = 2); external name ''A.B'';',
  17053. ' end;',
  17054. 'var',
  17055. ' A: texta;',
  17056. 'begin',
  17057. ' a:=texta.new;',
  17058. ' a:=texta(texta.new);',
  17059. ' a:=texta.new();',
  17060. ' a:=texta.new(1);',
  17061. ' with texta do begin',
  17062. ' a:=new;',
  17063. ' a:=new();',
  17064. ' a:=new(2);',
  17065. ' end;',
  17066. ' a:=test1.texta.new;',
  17067. ' a:=test1.texta.new();',
  17068. ' a:=test1.texta.new(3);',
  17069. '']);
  17070. ConvertProgram;
  17071. CheckSource('TestExternalClass_NewExtName',
  17072. LinesToStr([ // statements
  17073. 'this.A = null;',
  17074. '']),
  17075. LinesToStr([ // $mod.$main
  17076. '$mod.A = new Other();',
  17077. '$mod.A = new Other();',
  17078. '$mod.A = new Other();',
  17079. '$mod.A = new A.B(1,2);',
  17080. '$mod.A = new Other();',
  17081. '$mod.A = new Other();',
  17082. '$mod.A = new A.B(2,2);',
  17083. '$mod.A = new Other();',
  17084. '$mod.A = new Other();',
  17085. '$mod.A = new A.B(3,2);',
  17086. '']));
  17087. end;
  17088. procedure TTestModule.TestExternalClass_Constructor;
  17089. begin
  17090. StartProgram(false);
  17091. Add([
  17092. '{$modeswitch externalclass}',
  17093. 'type',
  17094. ' TExtA = class external name ''ExtA''',
  17095. ' constructor Create;',
  17096. ' constructor Create(i: longint; j: longint = 2);',
  17097. ' end;',
  17098. 'var',
  17099. ' A: texta;',
  17100. 'begin',
  17101. ' a:=texta.create;',
  17102. ' a:=texta(texta.create);',
  17103. ' a:=texta.create();',
  17104. ' a:=texta.create(1);',
  17105. ' with texta do begin',
  17106. ' a:=create;',
  17107. ' a:=create();',
  17108. ' a:=create(2);',
  17109. ' end;',
  17110. ' a:=test1.texta.create;',
  17111. ' a:=test1.texta.create();',
  17112. ' a:=test1.texta.create(3);',
  17113. '']);
  17114. ConvertProgram;
  17115. CheckSource('TestExternalClass_Constructor',
  17116. LinesToStr([ // statements
  17117. 'this.A = null;',
  17118. '']),
  17119. LinesToStr([ // $mod.$main
  17120. '$mod.A = new ExtA.Create();',
  17121. '$mod.A = new ExtA.Create();',
  17122. '$mod.A = new ExtA.Create();',
  17123. '$mod.A = new ExtA.Create(1,2);',
  17124. '$mod.A = new ExtA.Create();',
  17125. '$mod.A = new ExtA.Create();',
  17126. '$mod.A = new ExtA.Create(2,2);',
  17127. '$mod.A = new ExtA.Create();',
  17128. '$mod.A = new ExtA.Create();',
  17129. '$mod.A = new ExtA.Create(3,2);',
  17130. '']));
  17131. end;
  17132. procedure TTestModule.TestExternalClass_ConstructorBrackets;
  17133. begin
  17134. StartProgram(false);
  17135. Add([
  17136. '{$modeswitch externalclass}',
  17137. 'type',
  17138. ' TExtA = class external name ''ExtA''',
  17139. ' constructor Create; external name ''{}'';',
  17140. ' end;',
  17141. 'var',
  17142. ' A: texta;',
  17143. 'begin',
  17144. ' a:=texta.create;',
  17145. ' a:=texta(texta.create);',
  17146. ' a:=texta.create();',
  17147. ' with texta do begin',
  17148. ' a:=create;',
  17149. ' a:=create();',
  17150. ' end;',
  17151. ' a:=test1.texta.create;',
  17152. ' a:=test1.texta.create();',
  17153. '']);
  17154. ConvertProgram;
  17155. CheckSource('TestExternalClass_ConstructorBrackets',
  17156. LinesToStr([ // statements
  17157. 'this.A = null;',
  17158. '']),
  17159. LinesToStr([ // $mod.$main
  17160. '$mod.A = {};',
  17161. '$mod.A = {};',
  17162. '$mod.A = {};',
  17163. '$mod.A = {};',
  17164. '$mod.A = {};',
  17165. '$mod.A = {};',
  17166. '$mod.A = {};',
  17167. '']));
  17168. end;
  17169. procedure TTestModule.TestExternalClass_LocalConstSameName;
  17170. begin
  17171. StartProgram(false);
  17172. Add('{$modeswitch externalclass}');
  17173. Add('type');
  17174. Add(' TExtA = class external name ''ExtA''');
  17175. Add(' constructor New;');
  17176. Add(' end;');
  17177. Add('function DoIt: longint;');
  17178. Add('const ExtA: longint = 3;');
  17179. Add('begin');
  17180. Add(' Result:=ExtA;');
  17181. Add('end;');
  17182. Add('var');
  17183. Add(' A: texta;');
  17184. Add('begin');
  17185. Add(' a:=texta.new;');
  17186. ConvertProgram;
  17187. CheckSource('TestExternalClass_LocalConstSameName',
  17188. LinesToStr([ // statements
  17189. 'var ExtA$1 = 3;',
  17190. 'this.DoIt = function () {',
  17191. ' var Result = 0;',
  17192. ' Result = ExtA$1;',
  17193. ' return Result;',
  17194. '};',
  17195. 'this.A = null;',
  17196. '']),
  17197. LinesToStr([ // $mod.$main
  17198. '$mod.A = new ExtA();',
  17199. '']));
  17200. end;
  17201. procedure TTestModule.TestExternalClass_ReintroduceOverload;
  17202. begin
  17203. StartProgram(false);
  17204. Add('{$modeswitch externalclass}');
  17205. Add('type');
  17206. Add(' TExtA = class external name ''ExtA''');
  17207. Add(' procedure DoIt;');
  17208. Add(' end;');
  17209. Add(' TMyA = class(TExtA)');
  17210. Add(' procedure DoIt;');
  17211. Add(' end;');
  17212. Add('procedure TMyA.DoIt; begin end;');
  17213. Add('begin');
  17214. ConvertProgram;
  17215. CheckSource('TestExternalClass_ReintroduceOverload',
  17216. LinesToStr([ // statements
  17217. 'rtl.createClassExt(this, "TMyA", ExtA, "", function () {',
  17218. ' this.$init = function () {',
  17219. ' };',
  17220. ' this.$final = function () {',
  17221. ' };',
  17222. ' this.DoIt$1 = function () {',
  17223. ' };',
  17224. '});',
  17225. '']),
  17226. LinesToStr([ // $mod.$main
  17227. '']));
  17228. end;
  17229. procedure TTestModule.TestExternalClass_Inherited;
  17230. begin
  17231. StartProgram(false);
  17232. Add('{$modeswitch externalclass}');
  17233. Add('type');
  17234. Add(' TExtA = class external name ''ExtA''');
  17235. Add(' procedure DoIt(i: longint = 1); virtual;');
  17236. Add(' procedure DoSome(j: longint = 2);');
  17237. Add(' end;');
  17238. Add(' TExtB = class external name ''ExtB''(TExtA)');
  17239. Add(' end;');
  17240. Add(' TMyC = class(TExtB)');
  17241. Add(' procedure DoIt(i: longint = 1); override;');
  17242. Add(' procedure DoSome(j: longint = 2); reintroduce;');
  17243. Add(' end;');
  17244. Add('procedure TMyC.DoIt(i: longint);');
  17245. Add('begin');
  17246. Add(' inherited;');
  17247. Add(' inherited DoIt;');
  17248. Add(' inherited DoIt();');
  17249. Add(' inherited DoIt(3);');
  17250. Add(' inherited DoSome;');
  17251. Add(' inherited DoSome();');
  17252. Add(' inherited DoSome(4);');
  17253. Add('end;');
  17254. Add('procedure TMyC.DoSome(j: longint);');
  17255. Add('begin');
  17256. Add(' inherited;');
  17257. Add('end;');
  17258. Add('begin');
  17259. ConvertProgram;
  17260. CheckSource('TestExternalClass_ReintroduceOverload',
  17261. LinesToStr([ // statements
  17262. 'rtl.createClassExt(this, "TMyC", ExtB, "", function () {',
  17263. ' this.$init = function () {',
  17264. ' };',
  17265. ' this.$final = function () {',
  17266. ' };',
  17267. ' this.DoIt = function (i) {',
  17268. ' ExtB.DoIt.apply(this, arguments);',
  17269. ' ExtB.DoIt.call(this, 1);',
  17270. ' ExtB.DoIt.call(this, 1);',
  17271. ' ExtB.DoIt.call(this, 3);',
  17272. ' ExtB.DoSome.call(this, 2);',
  17273. ' ExtB.DoSome.call(this, 2);',
  17274. ' ExtB.DoSome.call(this, 4);',
  17275. ' };',
  17276. ' this.DoSome$1 = function (j) {',
  17277. ' ExtB.DoSome.apply(this, arguments);',
  17278. ' };',
  17279. '});',
  17280. '']),
  17281. LinesToStr([ // $mod.$main
  17282. '']));
  17283. end;
  17284. procedure TTestModule.TestExternalClass_PascalAncestorFail;
  17285. begin
  17286. StartProgram(false);
  17287. Add('{$modeswitch externalclass}');
  17288. Add('type');
  17289. Add(' TObject = class');
  17290. Add(' end;');
  17291. Add(' TExtA = class external name ''ExtA''(TObject)');
  17292. Add(' end;');
  17293. Add('begin');
  17294. SetExpectedPasResolverError('Ancestor "TObject" is not external',nAncestorIsNotExternal);
  17295. ConvertProgram;
  17296. end;
  17297. procedure TTestModule.TestExternalClass_NewInstance;
  17298. begin
  17299. StartProgram(false);
  17300. Add('{$modeswitch externalclass}');
  17301. Add('type');
  17302. Add(' TExtA = class external name ''ExtA''');
  17303. Add(' end;');
  17304. Add(' TMyB = class(TExtA)');
  17305. Add(' protected');
  17306. Add(' class function NewInstance(fnname: string; const paramarray): TMyB; virtual;');
  17307. Add(' end;');
  17308. Add('class function TMyB.NewInstance(fnname: string; const paramarray): TMyB;');
  17309. Add('begin end;');
  17310. Add('begin');
  17311. ConvertProgram;
  17312. CheckSource('TestExternalClass_NewInstance',
  17313. LinesToStr([ // statements
  17314. 'rtl.createClassExt(this, "TMyB", ExtA, "NewInstance", function () {',
  17315. ' this.$init = function () {',
  17316. ' };',
  17317. ' this.$final = function () {',
  17318. ' };',
  17319. ' this.NewInstance = function (fnname, paramarray) {',
  17320. ' var Result = null;',
  17321. ' return Result;',
  17322. ' };',
  17323. '});',
  17324. '']),
  17325. LinesToStr([ // $mod.$main
  17326. '']));
  17327. end;
  17328. procedure TTestModule.TestExternalClass_NewInstance_NonVirtualFail;
  17329. begin
  17330. StartProgram(false);
  17331. Add('{$modeswitch externalclass}');
  17332. Add('type');
  17333. Add(' TExtA = class external name ''ExtA''');
  17334. Add(' end;');
  17335. Add(' TMyB = class(TExtA)');
  17336. Add(' protected');
  17337. Add(' class function NewInstance(fnname: string; const paramarray): TMyB;');
  17338. Add(' end;');
  17339. Add('class function TMyB.NewInstance(fnname: string; const paramarray): TMyB;');
  17340. Add('begin end;');
  17341. Add('begin');
  17342. SetExpectedPasResolverError(sNewInstanceFunctionMustBeVirtual,nNewInstanceFunctionMustBeVirtual);
  17343. ConvertProgram;
  17344. end;
  17345. procedure TTestModule.TestExternalClass_NewInstance_FirstParamNotString_Fail;
  17346. begin
  17347. StartProgram(false);
  17348. Add('{$modeswitch externalclass}');
  17349. Add('type');
  17350. Add(' TExtA = class external name ''ExtA''');
  17351. Add(' end;');
  17352. Add(' TMyB = class(TExtA)');
  17353. Add(' protected');
  17354. Add(' class function NewInstance(fnname: longint; const paramarray): TMyB; virtual;');
  17355. Add(' end;');
  17356. Add('class function TMyB.NewInstance(fnname: longint; const paramarray): TMyB;');
  17357. Add('begin end;');
  17358. Add('begin');
  17359. SetExpectedPasResolverError('Incompatible type arg no. 1: Got "Longint", expected "String"',
  17360. nIncompatibleTypeArgNo);
  17361. ConvertProgram;
  17362. end;
  17363. procedure TTestModule.TestExternalClass_NewInstance_SecondParamTyped_Fail;
  17364. begin
  17365. StartProgram(false);
  17366. Add('{$modeswitch externalclass}');
  17367. Add('type');
  17368. Add(' TExtA = class external name ''ExtA''');
  17369. Add(' end;');
  17370. Add(' TMyB = class(TExtA)');
  17371. Add(' protected');
  17372. Add(' class function NewInstance(fnname: string; const paramarray: string): TMyB; virtual;');
  17373. Add(' end;');
  17374. Add('class function TMyB.NewInstance(fnname: string; const paramarray: string): TMyB;');
  17375. Add('begin end;');
  17376. Add('begin');
  17377. SetExpectedPasResolverError('Incompatible type arg no. 2: Got "type", expected "untyped"',
  17378. nIncompatibleTypeArgNo);
  17379. ConvertProgram;
  17380. end;
  17381. procedure TTestModule.TestExternalClass_JSFunctionPasDescendant;
  17382. begin
  17383. StartProgram(false);
  17384. Add([
  17385. '{$modeswitch externalclass}',
  17386. 'type',
  17387. ' TJSFunction = class external name ''Function''',
  17388. ' end;',
  17389. ' TExtA = class external name ''ExtA''(TJSFunction)',
  17390. ' constructor New(w: word);',
  17391. ' end;',
  17392. ' TBird = class (TExtA)',
  17393. ' public',
  17394. ' Size: word;',
  17395. ' class var Legs: word;',
  17396. ' constructor Create(a: word);',
  17397. ' end;',
  17398. ' TEagle = class (TBird)',
  17399. ' public',
  17400. ' constructor Create(b: word); reintroduce;',
  17401. ' end;',
  17402. 'constructor TBird.Create(a: word);',
  17403. 'begin',
  17404. ' inherited;', // silently ignored
  17405. ' inherited New(a);', // this.$func(a)
  17406. 'end;',
  17407. 'constructor TEagle.Create(b: word);',
  17408. 'begin',
  17409. ' inherited Create(b);',
  17410. 'end;',
  17411. 'var',
  17412. ' Bird: TBird;',
  17413. ' Eagle: TEagle;',
  17414. 'begin',
  17415. ' Bird:=TBird.Create(3);',
  17416. ' Eagle:=TEagle.Create(4);',
  17417. ' Bird.Size:=Bird.Size+5;',
  17418. ' Bird.Legs:=Bird.Legs+6;',
  17419. ' Eagle.Size:=Eagle.Size+5;',
  17420. ' Eagle.Legs:=Eagle.Legs+6;',
  17421. '']);
  17422. ConvertProgram;
  17423. CheckSource('TestExternalClass_JSFunctionPasDescendant',
  17424. LinesToStr([ // statements
  17425. 'rtl.createClassExt(this, "TBird", ExtA, "", function () {',
  17426. ' this.Legs = 0;',
  17427. ' this.$init = function () {',
  17428. ' this.Size = 0;',
  17429. ' };',
  17430. ' this.$final = function () {',
  17431. ' };',
  17432. ' this.Create = function (a) {',
  17433. ' this.$ancestorfunc(a);',
  17434. ' return this;',
  17435. ' };',
  17436. '});',
  17437. 'rtl.createClassExt(this, "TEagle", this.TBird, "", function () {',
  17438. ' this.Create$1 = function (b) {',
  17439. ' $mod.TBird.Create.call(this, b);',
  17440. ' return this;',
  17441. ' };',
  17442. '});',
  17443. 'this.Bird = null;',
  17444. 'this.Eagle = null;',
  17445. '']),
  17446. LinesToStr([ // $mod.$main
  17447. '$mod.Bird = $mod.TBird.$create("Create", [3]);',
  17448. '$mod.Eagle = $mod.TEagle.$create("Create$1", [4]);',
  17449. '$mod.Bird.Size = $mod.Bird.Size + 5;',
  17450. '$mod.TBird.Legs = $mod.Bird.Legs + 6;',
  17451. '$mod.Eagle.Size = $mod.Eagle.Size + 5;',
  17452. '$mod.TBird.Legs = $mod.Eagle.Legs + 6;',
  17453. '']));
  17454. end;
  17455. procedure TTestModule.TestExternalClass_PascalProperty;
  17456. begin
  17457. StartProgram(false);
  17458. Add('{$modeswitch externalclass}');
  17459. Add('type');
  17460. Add(' TJSElement = class;');
  17461. Add(' TJSNotifyEvent = procedure(Sender: TJSElement) of object;');
  17462. Add(' TJSElement = class external name ''ExtA''');
  17463. Add(' end;');
  17464. Add(' TControl = class(TJSElement)');
  17465. Add(' private');
  17466. Add(' FOnClick: TJSNotifyEvent;');
  17467. Add(' property OnClick: TJSNotifyEvent read FOnClick write FOnClick;');
  17468. Add(' procedure Click(Sender: TJSElement);');
  17469. Add(' end;');
  17470. Add('procedure TControl.Click(Sender: TJSElement);');
  17471. Add('begin');
  17472. Add(' OnClick(Self);');
  17473. Add('end;');
  17474. Add('var');
  17475. Add(' Ctrl: TControl;');
  17476. Add('begin');
  17477. Add(' Ctrl.OnClick:[email protected];');
  17478. Add(' Ctrl.OnClick(Ctrl);');
  17479. ConvertProgram;
  17480. CheckSource('TestExternalClass_PascalProperty',
  17481. LinesToStr([ // statements
  17482. 'rtl.createClassExt(this, "TControl", ExtA, "", function () {',
  17483. ' this.$init = function () {',
  17484. ' this.FOnClick = null;',
  17485. ' };',
  17486. ' this.$final = function () {',
  17487. ' this.FOnClick = undefined;',
  17488. ' };',
  17489. ' this.Click = function (Sender) {',
  17490. ' this.FOnClick(this);',
  17491. ' };',
  17492. '});',
  17493. 'this.Ctrl = null;',
  17494. '']),
  17495. LinesToStr([ // $mod.$main
  17496. '$mod.Ctrl.FOnClick = rtl.createCallback($mod.Ctrl, "Click");',
  17497. '$mod.Ctrl.FOnClick($mod.Ctrl);',
  17498. '']));
  17499. end;
  17500. procedure TTestModule.TestExternalClass_TypeCastToRootClass;
  17501. begin
  17502. StartProgram(false);
  17503. Add([
  17504. '{$modeswitch externalclass}',
  17505. 'type',
  17506. ' IUnknown = interface end;',
  17507. ' TObject = class',
  17508. ' end;',
  17509. ' TChild = class',
  17510. ' end;',
  17511. ' TExtRootA = class external name ''ExtRootA''',
  17512. ' end;',
  17513. ' TExtChildA = class external name ''ExtChildA''(TExtRootA)',
  17514. ' end;',
  17515. ' TExtRootB = class external name ''ExtRootB''',
  17516. ' end;',
  17517. ' TExtChildB = class external name ''ExtChildB''(TExtRootB)',
  17518. ' end;',
  17519. 'var',
  17520. ' Obj: TObject;',
  17521. ' Child: TChild;',
  17522. ' RootA: TExtRootA;',
  17523. ' ChildA: TExtChildA;',
  17524. ' RootB: TExtRootB;',
  17525. ' ChildB: TExtChildB;',
  17526. ' i: IUnknown;',
  17527. 'begin',
  17528. ' obj:=tobject(roota);',
  17529. ' obj:=tobject(childa);',
  17530. ' child:=tchild(tobject(roota));',
  17531. ' roota:=textroota(obj);',
  17532. ' roota:=textroota(child);',
  17533. ' roota:=textroota(rootb);',
  17534. ' roota:=textroota(childb);',
  17535. ' childa:=textchilda(textroota(obj));',
  17536. ' roota:=TExtRootA(i)',
  17537. '']);
  17538. ConvertProgram;
  17539. CheckSource('TestExternalClass_TypeCastToRootClass',
  17540. LinesToStr([ // statements
  17541. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  17542. 'rtl.createClass(this, "TObject", null, function () {',
  17543. ' this.$init = function () {',
  17544. ' };',
  17545. ' this.$final = function () {',
  17546. ' };',
  17547. '});',
  17548. 'rtl.createClass(this, "TChild", this.TObject, function () {',
  17549. '});',
  17550. 'this.Obj = null;',
  17551. 'this.Child = null;',
  17552. 'this.RootA = null;',
  17553. 'this.ChildA = null;',
  17554. 'this.RootB = null;',
  17555. 'this.ChildB = null;',
  17556. 'this.i = null;',
  17557. '']),
  17558. LinesToStr([ // $mod.$main
  17559. '$mod.Obj = $mod.RootA;',
  17560. '$mod.Obj = $mod.ChildA;',
  17561. '$mod.Child = $mod.RootA;',
  17562. '$mod.RootA = $mod.Obj;',
  17563. '$mod.RootA = $mod.Child;',
  17564. '$mod.RootA = $mod.RootB;',
  17565. '$mod.RootA = $mod.ChildB;',
  17566. '$mod.ChildA = $mod.Obj;',
  17567. '$mod.RootA = $mod.i;',
  17568. '']));
  17569. end;
  17570. procedure TTestModule.TestExternalClass_TypeCastToJSObject;
  17571. begin
  17572. StartProgram(false);
  17573. Add([
  17574. '{$modeswitch externalclass}',
  17575. 'type',
  17576. ' IUnknown = interface end;',
  17577. ' IBird = interface(IUnknown) end;',
  17578. ' TClass = class of TObject;',
  17579. ' TObject = class',
  17580. ' end;',
  17581. ' TChild = class',
  17582. ' end;',
  17583. ' TJSObject = class external name ''Object''',
  17584. ' end;',
  17585. ' TRec = record end;',
  17586. 'var',
  17587. ' Obj: TObject;',
  17588. ' Child: TChild;',
  17589. ' i: IUnknown;',
  17590. ' Bird: IBird;',
  17591. ' j: TJSObject;',
  17592. ' r: TRec;',
  17593. ' c: TClass;',
  17594. 'begin',
  17595. ' j:=tjsobject(IUnknown);',
  17596. ' j:=tjsobject(IBird);',
  17597. ' j:=tjsobject(TObject);',
  17598. ' j:=tjsobject(TChild);',
  17599. ' j:=tjsobject(TRec);',
  17600. ' j:=tjsobject(Obj);',
  17601. ' j:=tjsobject(Child);',
  17602. ' j:=tjsobject(i);',
  17603. ' j:=tjsobject(Bird);',
  17604. ' j:=tjsobject(r);',
  17605. ' j:=tjsobject(c);',
  17606. '']);
  17607. ConvertProgram;
  17608. CheckSource('TestExternalClass_TypeCastToJSObject',
  17609. LinesToStr([ // statements
  17610. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  17611. 'rtl.createInterface(this, "IBird", "{4B0D080B-C0F6-396E-AE88-000B87785074}", [], this.IUnknown);',
  17612. 'rtl.createClass(this, "TObject", null, function () {',
  17613. ' this.$init = function () {',
  17614. ' };',
  17615. ' this.$final = function () {',
  17616. ' };',
  17617. '});',
  17618. 'rtl.createClass(this, "TChild", this.TObject, function () {',
  17619. '});',
  17620. 'rtl.recNewT(this, "TRec", function () {',
  17621. ' this.$eq = function (b) {',
  17622. ' return true;',
  17623. ' };',
  17624. ' this.$assign = function (s) {',
  17625. ' return this;',
  17626. ' };',
  17627. '});',
  17628. 'this.Obj = null;',
  17629. 'this.Child = null;',
  17630. 'this.i = null;',
  17631. 'this.Bird = null;',
  17632. 'this.j = null;',
  17633. 'this.r = this.TRec.$new();',
  17634. 'this.c = null;',
  17635. '']),
  17636. LinesToStr([ // $mod.$main
  17637. '$mod.j = $mod.IUnknown;',
  17638. '$mod.j = $mod.IBird;',
  17639. '$mod.j = $mod.TObject;',
  17640. '$mod.j = $mod.TChild;',
  17641. '$mod.j = $mod.TRec;',
  17642. '$mod.j = $mod.Obj;',
  17643. '$mod.j = $mod.Child;',
  17644. '$mod.j = $mod.i;',
  17645. '$mod.j = $mod.Bird;',
  17646. '$mod.j = $mod.r;',
  17647. '$mod.j = $mod.c;',
  17648. '']));
  17649. end;
  17650. procedure TTestModule.TestExternalClass_TypeCastStringToExternalString;
  17651. begin
  17652. StartProgram(false);
  17653. Add('{$modeswitch externalclass}');
  17654. Add('type');
  17655. Add(' TJSString = class external name ''String''');
  17656. Add(' class function fromCharCode() : string; varargs;');
  17657. Add(' function anchor(const aName : string) : string;');
  17658. Add(' end;');
  17659. Add('var');
  17660. Add(' s: string;');
  17661. Add('begin');
  17662. Add(' s:=TJSString.fromCharCode(65,66);');
  17663. Add(' s:=TJSString(s).anchor(s);');
  17664. Add(' s:=TJSString(''foo'').anchor(s);');
  17665. ConvertProgram;
  17666. CheckSource('TestExternalClass_TypeCastStringToExternalString',
  17667. LinesToStr([ // statements
  17668. 'this.s = "";',
  17669. '']),
  17670. LinesToStr([ // $mod.$main
  17671. '$mod.s = String.fromCharCode(65, 66);',
  17672. '$mod.s = $mod.s.anchor($mod.s);',
  17673. '$mod.s = "foo".anchor($mod.s);',
  17674. '']));
  17675. end;
  17676. procedure TTestModule.TestExternalClass_TypeCastToJSFunction;
  17677. begin
  17678. StartProgram(false);
  17679. Add([
  17680. '{$modeswitch externalclass}',
  17681. 'type',
  17682. ' TJSObject = class external name ''Object'' end;',
  17683. ' TJSFunction = class external name ''Function''',
  17684. ' function bind(thisArg: TJSObject): TJSFunction; varargs;',
  17685. ' function call(thisArg: TJSObject): JSValue; varargs;',
  17686. ' end;',
  17687. ' TObject = class',
  17688. ' procedure DoIt(i: longint);',
  17689. ' end;',
  17690. ' TFuncInt = function(o: TObject): longint;',
  17691. 'function GetIt(o: TObject): longint;',
  17692. ' procedure Sub; begin end;',
  17693. 'var',
  17694. ' f: TJSFunction;',
  17695. ' fi: TFuncInt;',
  17696. 'begin',
  17697. ' fi:=TFuncInt(f);',
  17698. ' f:=TJSFunction(fi);',
  17699. ' f:=TJSFunction(@GetIt);',
  17700. ' f:=TJSFunction(@GetIt).bind(nil,3);',
  17701. ' f:=TJSFunction(@Sub);',
  17702. ' f:=TJSFunction(@o.doit);',
  17703. ' f:=TJSFunction(fi).bind(nil,4)',
  17704. 'end;',
  17705. 'procedure TObject.DoIt(i: longint);',
  17706. ' procedure Sub; begin end;',
  17707. 'var f: TJSFunction;',
  17708. 'begin',
  17709. ' f:=TJSFunction(@DoIt);',
  17710. ' f:=TJSFunction(@DoIt).bind(nil,13);',
  17711. ' f:=TJSFunction(@Sub);',
  17712. ' f:=TJSFunction(@GetIt);',
  17713. 'end;',
  17714. 'begin']);
  17715. ConvertProgram;
  17716. CheckSource('TestExternalClass_TypeCastToJSFunction',
  17717. LinesToStr([ // statements
  17718. 'rtl.createClass(this, "TObject", null, function () {',
  17719. ' this.$init = function () {',
  17720. ' };',
  17721. ' this.$final = function () {',
  17722. ' };',
  17723. ' this.DoIt = function (i) {',
  17724. ' var $Self = this;',
  17725. ' function Sub() {',
  17726. ' };',
  17727. ' var f = null;',
  17728. ' f = this.DoIt;',
  17729. ' f = this.DoIt.bind(null, 13);',
  17730. ' f = Sub;',
  17731. ' f = $mod.GetIt;',
  17732. ' };',
  17733. '});',
  17734. 'this.GetIt = function (o) {',
  17735. ' var Result = 0;',
  17736. ' function Sub() {',
  17737. ' };',
  17738. ' var f = null;',
  17739. ' var fi = null;',
  17740. ' fi = f;',
  17741. ' f = fi;',
  17742. ' f = $mod.GetIt;',
  17743. ' f = $mod.GetIt.bind(null, 3);',
  17744. ' f = Sub;',
  17745. ' f = $mod.TObject.DoIt;',
  17746. ' f = fi.bind(null, 4);',
  17747. ' return Result;',
  17748. '};',
  17749. '']),
  17750. LinesToStr([ // $mod.$main
  17751. '']));
  17752. end;
  17753. procedure TTestModule.TestExternalClass_TypeCastDelphiUnrelated;
  17754. begin
  17755. StartProgram(false);
  17756. Add([
  17757. '{$mode delphi}',
  17758. '{$modeswitch externalclass}',
  17759. 'type',
  17760. ' TJSObject = class external name ''Object'' end;',
  17761. ' TJSWindow = class external name ''Window''(TJSObject)',
  17762. ' procedure Open;',
  17763. ' end;',
  17764. ' TJSEventTarget = class external name ''Event''(TJSObject)',
  17765. ' procedure Execute;',
  17766. ' end;',
  17767. 'procedure Fly;',
  17768. 'var',
  17769. ' w: TJSWindow;',
  17770. ' e: TJSEventTarget;',
  17771. 'begin',
  17772. ' w:=TJSWindow(e);',
  17773. ' e:=TJSEventTarget(w);',
  17774. 'end;',
  17775. 'begin']);
  17776. ConvertProgram;
  17777. CheckSource('TestExternalClass_TypeCastDelphiUnrelated',
  17778. LinesToStr([ // statements
  17779. 'this.Fly = function () {',
  17780. ' var w = null;',
  17781. ' var e = null;',
  17782. ' w = e;',
  17783. ' e = w;',
  17784. '};',
  17785. '']),
  17786. LinesToStr([ // $mod.$main
  17787. '']));
  17788. end;
  17789. procedure TTestModule.TestExternalClass_CallClassFunctionOfInstanceFail;
  17790. begin
  17791. StartProgram(false);
  17792. Add('{$modeswitch externalclass}');
  17793. Add('type');
  17794. Add(' TJSString = class external name ''String''');
  17795. Add(' class function fromCharCode() : string; varargs;');
  17796. Add(' end;');
  17797. Add('var');
  17798. Add(' s: string;');
  17799. Add(' sObj: TJSString;');
  17800. Add('begin');
  17801. Add(' s:=sObj.fromCharCode(65,66);');
  17802. SetExpectedPasResolverError('External class instance cannot access static class function fromCharCode',
  17803. nExternalClassInstanceCannotAccessStaticX);
  17804. ConvertProgram;
  17805. end;
  17806. procedure TTestModule.TestExternalClass_BracketAccessor;
  17807. begin
  17808. StartProgram(false);
  17809. Add([
  17810. '{$modeswitch externalclass}',
  17811. 'type',
  17812. ' TJSArray = class external name ''Array2''',
  17813. ' function GetItems(Index: longint): jsvalue; external name ''[]'';',
  17814. ' procedure SetItems(Index: longint; Value: jsvalue); external name ''[]'';',
  17815. ' property Items[Index: longint]: jsvalue read GetItems write SetItems; default;',
  17816. ' end;',
  17817. 'procedure DoIt(vI: JSValue; const vJ: jsvalue; var vK: jsvalue; out vL: jsvalue);',
  17818. 'begin end;',
  17819. 'var',
  17820. ' Arr: tjsarray;',
  17821. ' s: string;',
  17822. ' i: longint;',
  17823. ' v: jsvalue;',
  17824. 'begin',
  17825. ' v:=arr[0];',
  17826. ' v:=arr.items[1];',
  17827. ' arr[2]:=s;',
  17828. ' arr.items[3]:=s;',
  17829. ' arr[4]:=i;',
  17830. ' arr[5]:=arr[6];',
  17831. ' arr.items[7]:=arr.items[8];',
  17832. ' with arr do items[9]:=items[10];',
  17833. ' doit(arr[7],arr[8],arr[9],arr[10]);',
  17834. ' with arr do begin',
  17835. ' v:=GetItems(14);',
  17836. ' setitems(15,16);',
  17837. ' end;',
  17838. ' v:=test1.arr.items[17];',
  17839. ' test1.arr.items[18]:=v;',
  17840. '']);
  17841. ConvertProgram;
  17842. CheckSource('TestExternalClass_BracketAccessor',
  17843. LinesToStr([ // statements
  17844. 'this.DoIt = function (vI, vJ, vK, vL) {',
  17845. '};',
  17846. 'this.Arr = null;',
  17847. 'this.s = "";',
  17848. 'this.i = 0;',
  17849. 'this.v = undefined;',
  17850. '']),
  17851. LinesToStr([ // $mod.$main
  17852. '$mod.v = $mod.Arr[0];',
  17853. '$mod.v = $mod.Arr[1];',
  17854. '$mod.Arr[2] = $mod.s;',
  17855. '$mod.Arr[3] = $mod.s;',
  17856. '$mod.Arr[4] = $mod.i;',
  17857. '$mod.Arr[5] = $mod.Arr[6];',
  17858. '$mod.Arr[7] = $mod.Arr[8];',
  17859. 'var $with = $mod.Arr;',
  17860. '$with[9] = $with[10];',
  17861. '$mod.DoIt($mod.Arr[7], $mod.Arr[8], {',
  17862. ' a: 9,',
  17863. ' p: $mod.Arr,',
  17864. ' get: function () {',
  17865. ' return this.p[this.a];',
  17866. ' },',
  17867. ' set: function (v) {',
  17868. ' this.p[this.a] = v;',
  17869. ' }',
  17870. '}, {',
  17871. ' a: 10,',
  17872. ' p: $mod.Arr,',
  17873. ' get: function () {',
  17874. ' return this.p[this.a];',
  17875. ' },',
  17876. ' set: function (v) {',
  17877. ' this.p[this.a] = v;',
  17878. ' }',
  17879. '});',
  17880. 'var $with1 = $mod.Arr;',
  17881. '$mod.v = $with1[14];',
  17882. '$with1[15] = 16;',
  17883. '$mod.v = $mod.Arr[17];',
  17884. '$mod.Arr[18] = $mod.v;',
  17885. '']));
  17886. end;
  17887. procedure TTestModule.TestExternalClass_BracketAccessor_Call;
  17888. begin
  17889. StartProgram(false);
  17890. Add([
  17891. '{$modeswitch externalclass}',
  17892. 'type',
  17893. ' TJSArray = class external name ''Array2''',
  17894. ' function GetItems(Index: longint): jsvalue; external name ''[]'';',
  17895. ' procedure SetItems(Index: longint; Value: jsvalue); external name ''[]'';',
  17896. ' property Items[Index: longint]: jsvalue read GetItems write SetItems; default;',
  17897. ' end;',
  17898. ' TMyArr = class(TJSArray)',
  17899. ' procedure DoIt;',
  17900. ' end;',
  17901. 'procedure tmyarr.DoIt;',
  17902. 'begin',
  17903. ' Items[1]:=Items[2];',
  17904. ' SetItems(3,getItems(4));',
  17905. 'end;',
  17906. 'var',
  17907. ' Arr: tmyarr;',
  17908. ' s: string;',
  17909. ' i: longint;',
  17910. ' v: jsvalue;',
  17911. 'begin',
  17912. ' v:=arr[0];',
  17913. ' v:=arr.items[1];',
  17914. ' arr[2]:=s;',
  17915. ' arr.items[3]:=s;',
  17916. ' arr[4]:=i;',
  17917. ' arr[5]:=arr[6];',
  17918. ' arr.items[7]:=arr.items[8];',
  17919. ' with arr do items[9]:=items[10];',
  17920. ' with arr do begin',
  17921. ' v:=GetItems(14);',
  17922. ' setitems(15,16);',
  17923. ' end;',
  17924. '']);
  17925. ConvertProgram;
  17926. CheckSource('TestExternalClass_BracketAccessor_Call',
  17927. LinesToStr([ // statements
  17928. 'rtl.createClassExt(this, "TMyArr", Array2, "", function () {',
  17929. ' this.$init = function () {',
  17930. ' };',
  17931. ' this.$final = function () {',
  17932. ' };',
  17933. ' this.DoIt = function () {',
  17934. ' this[1] = this[2];',
  17935. ' this[3] = this[4];',
  17936. ' };',
  17937. '});',
  17938. 'this.Arr = null;',
  17939. 'this.s = "";',
  17940. 'this.i = 0;',
  17941. 'this.v = undefined;',
  17942. '']),
  17943. LinesToStr([ // $mod.$main
  17944. '$mod.v = $mod.Arr[0];',
  17945. '$mod.v = $mod.Arr[1];',
  17946. '$mod.Arr[2] = $mod.s;',
  17947. '$mod.Arr[3] = $mod.s;',
  17948. '$mod.Arr[4] = $mod.i;',
  17949. '$mod.Arr[5] = $mod.Arr[6];',
  17950. '$mod.Arr[7] = $mod.Arr[8];',
  17951. 'var $with = $mod.Arr;',
  17952. '$with[9] = $with[10];',
  17953. 'var $with1 = $mod.Arr;',
  17954. '$mod.v = $with1[14];',
  17955. '$with1[15] = 16;',
  17956. '']));
  17957. end;
  17958. procedure TTestModule.TestExternalClass_BracketAccessor_2ParamsFail;
  17959. begin
  17960. StartProgram(false);
  17961. Add('{$modeswitch externalclass}');
  17962. Add('type');
  17963. Add(' TJSArray = class external name ''Array2''');
  17964. Add(' function GetItems(Index1, Index2: longint): jsvalue; external name ''[]'';');
  17965. Add(' procedure SetItems(Index1, Index2: longint; Value: jsvalue); external name ''[]'';');
  17966. Add(' property Items[Index1, Index2: longint]: jsvalue read GetItems write SetItems; default;');
  17967. Add(' end;');
  17968. Add('begin');
  17969. SetExpectedPasResolverError(sBracketAccessorOfExternalClassMustHaveOneParameter,
  17970. nBracketAccessorOfExternalClassMustHaveOneParameter);
  17971. ConvertProgram;
  17972. end;
  17973. procedure TTestModule.TestExternalClass_BracketAccessor_ReadOnly;
  17974. begin
  17975. StartProgram(false);
  17976. Add('{$modeswitch externalclass}');
  17977. Add('type');
  17978. Add(' TJSArray = class external name ''Array2''');
  17979. Add(' function GetItems(Index: longint): jsvalue; external name ''[]'';');
  17980. Add(' property Items[Index: longint]: jsvalue read GetItems; default;');
  17981. Add(' end;');
  17982. Add('procedure DoIt(vI: JSValue; const vJ: jsvalue);');
  17983. Add('begin end;');
  17984. Add('var');
  17985. Add(' Arr: tjsarray;');
  17986. Add(' v: jsvalue;');
  17987. Add('begin');
  17988. Add(' v:=arr[0];');
  17989. Add(' v:=arr.items[1];');
  17990. Add(' with arr do v:=items[2];');
  17991. Add(' doit(arr[3],arr[4]);');
  17992. ConvertProgram;
  17993. CheckSource('TestExternalClass_BracketAccessor_ReadOnly',
  17994. LinesToStr([ // statements
  17995. 'this.DoIt = function (vI, vJ) {',
  17996. '};',
  17997. 'this.Arr = null;',
  17998. 'this.v = undefined;',
  17999. '']),
  18000. LinesToStr([ // $mod.$main
  18001. '$mod.v = $mod.Arr[0];',
  18002. '$mod.v = $mod.Arr[1];',
  18003. 'var $with = $mod.Arr;',
  18004. '$mod.v = $with[2];',
  18005. '$mod.DoIt($mod.Arr[3], $mod.Arr[4]);',
  18006. '']));
  18007. end;
  18008. procedure TTestModule.TestExternalClass_BracketAccessor_WriteOnly;
  18009. begin
  18010. StartProgram(false);
  18011. Add('{$modeswitch externalclass}');
  18012. Add('type');
  18013. Add(' TJSArray = class external name ''Array2''');
  18014. Add(' procedure SetItems(Index: longint; Value: jsvalue); external name ''[]'';');
  18015. Add(' property Items[Index: longint]: jsvalue write SetItems; default;');
  18016. Add(' end;');
  18017. Add('var');
  18018. Add(' Arr: tjsarray;');
  18019. Add(' s: string;');
  18020. Add(' i: longint;');
  18021. Add(' v: jsvalue;');
  18022. Add('begin');
  18023. Add(' arr[2]:=s;');
  18024. Add(' arr.items[3]:=s;');
  18025. Add(' arr[4]:=i;');
  18026. Add(' with arr do items[5]:=i;');
  18027. ConvertProgram;
  18028. CheckSource('TestExternalClass_BracketAccessor_WriteOnly',
  18029. LinesToStr([ // statements
  18030. 'this.Arr = null;',
  18031. 'this.s = "";',
  18032. 'this.i = 0;',
  18033. 'this.v = undefined;',
  18034. '']),
  18035. LinesToStr([ // $mod.$main
  18036. '$mod.Arr[2] = $mod.s;',
  18037. '$mod.Arr[3] = $mod.s;',
  18038. '$mod.Arr[4] = $mod.i;',
  18039. 'var $with = $mod.Arr;',
  18040. '$with[5] = $mod.i;',
  18041. '']));
  18042. end;
  18043. procedure TTestModule.TestExternalClass_BracketAccessor_MultiType;
  18044. begin
  18045. StartProgram(false);
  18046. Add('{$modeswitch externalclass}');
  18047. Add('type');
  18048. Add(' TJSArray = class external name ''Array2''');
  18049. Add(' procedure SetItems(Index: longint; Value: jsvalue); external name ''[]'';');
  18050. Add(' property Items[Index: longint]: jsvalue write SetItems; default;');
  18051. Add(' procedure SetNumbers(Index: longint; Value: longint); external name ''[]'';');
  18052. Add(' property Numbers[Index: longint]: longint write SetNumbers;');
  18053. Add(' end;');
  18054. Add('var');
  18055. Add(' Arr: tjsarray;');
  18056. Add(' s: string;');
  18057. Add(' i: longint;');
  18058. Add(' v: jsvalue;');
  18059. Add('begin');
  18060. Add(' arr[2]:=s;');
  18061. Add(' arr.items[3]:=s;');
  18062. Add(' arr.numbers[4]:=i;');
  18063. Add(' with arr do items[5]:=i;');
  18064. Add(' with arr do numbers[6]:=i;');
  18065. ConvertProgram;
  18066. CheckSource('TestExternalClass_BracketAccessor_MultiType',
  18067. LinesToStr([ // statements
  18068. 'this.Arr = null;',
  18069. 'this.s = "";',
  18070. 'this.i = 0;',
  18071. 'this.v = undefined;',
  18072. '']),
  18073. LinesToStr([ // $mod.$main
  18074. '$mod.Arr[2] = $mod.s;',
  18075. '$mod.Arr[3] = $mod.s;',
  18076. '$mod.Arr[4] = $mod.i;',
  18077. 'var $with = $mod.Arr;',
  18078. '$with[5] = $mod.i;',
  18079. 'var $with1 = $mod.Arr;',
  18080. '$with1[6] = $mod.i;',
  18081. '']));
  18082. end;
  18083. procedure TTestModule.TestExternalClass_BracketAccessor_Index;
  18084. begin
  18085. StartProgram(false);
  18086. Add('{$modeswitch externalclass}');
  18087. Add('type');
  18088. Add(' TJSArray = class external name ''Array2''');
  18089. Add(' function GetItems(Index: longint): jsvalue; external name ''[]'';');
  18090. Add(' procedure SetItems(Index: longint; Value: jsvalue); external name ''[]'';');
  18091. Add(' property Items[Index: longint]: jsvalue read GetItems write SetItems; default;');
  18092. Add(' end;');
  18093. Add('var');
  18094. Add(' Arr: tjsarray;');
  18095. Add(' i: longint;');
  18096. Add(' IntArr: array of longint;');
  18097. Add(' v: jsvalue;');
  18098. Add('begin');
  18099. Add(' v:=arr.items[i];');
  18100. Add(' arr[longint(v)]:=arr.items[intarr[0]];');
  18101. Add(' arr.items[intarr[1]]:=arr[IntArr[2]];');
  18102. ConvertProgram;
  18103. CheckSource('TestExternalClass_BracketAccessor_Index',
  18104. LinesToStr([ // statements
  18105. 'this.Arr = null;',
  18106. 'this.i = 0;',
  18107. 'this.IntArr = [];',
  18108. 'this.v = undefined;',
  18109. '']),
  18110. LinesToStr([ // $mod.$main
  18111. '$mod.v = $mod.Arr[$mod.i];',
  18112. '$mod.Arr[rtl.trunc($mod.v)] = $mod.Arr[$mod.IntArr[0]];',
  18113. '$mod.Arr[$mod.IntArr[1]] = $mod.Arr[$mod.IntArr[2]];',
  18114. '']));
  18115. end;
  18116. procedure TTestModule.TestExternalClass_ForInJSObject;
  18117. begin
  18118. StartProgram(false);
  18119. Add([
  18120. '{$modeswitch externalclass}',
  18121. 'type',
  18122. ' TJSObject = class external name ''Object''',
  18123. ' end;',
  18124. 'var',
  18125. ' o: TJSObject;',
  18126. ' key: string;',
  18127. 'begin',
  18128. ' for key in o do',
  18129. ' if key=''abc'' then ;',
  18130. '']);
  18131. ConvertProgram;
  18132. CheckSource('TestExternalClass_ForInJSObject',
  18133. LinesToStr([ // statements
  18134. 'this.o = null;',
  18135. 'this.key = "";',
  18136. '']),
  18137. LinesToStr([ // $mod.$main
  18138. 'for ($mod.key in $mod.o) if ($mod.key === "abc") ;',
  18139. '']));
  18140. end;
  18141. procedure TTestModule.TestExternalClass_ForInJSArray;
  18142. begin
  18143. StartProgram(false);
  18144. Add([
  18145. '{$modeswitch externalclass}',
  18146. 'type',
  18147. ' TJSInt8Array = class external name ''Int8Array''',
  18148. ' private',
  18149. ' flength: NativeInt external name ''length'';',
  18150. ' function getValue(Index: NativeInt): shortint; external name ''[]'';',
  18151. ' public',
  18152. ' property values[Index: NativeInt]: Shortint Read getValue; default;',
  18153. ' property Length: NativeInt read flength;',
  18154. ' end;',
  18155. 'var',
  18156. ' a: TJSInt8Array;',
  18157. ' value: shortint;',
  18158. 'begin',
  18159. ' for value in a do',
  18160. ' if value=3 then ;',
  18161. '']);
  18162. ConvertProgram;
  18163. CheckSource('TestExternalClass_ForInJSArray',
  18164. LinesToStr([ // statements
  18165. 'this.a = null;',
  18166. 'this.value = 0;',
  18167. '']),
  18168. LinesToStr([ // $mod.$main
  18169. 'for (var $in = $mod.a, $l = 0, $end = rtl.length($in) - 1; $l <= $end; $l++) {',
  18170. ' $mod.value = $in[$l];',
  18171. ' if ($mod.value === 3) ;',
  18172. '};',
  18173. '']));
  18174. end;
  18175. procedure TTestModule.TestExternalClass_IncompatibleArgDuplicateIdentifier;
  18176. begin
  18177. AddModuleWithIntfImplSrc('unit2.pas',
  18178. LinesToStr([
  18179. '{$modeswitch externalclass}',
  18180. 'type',
  18181. ' TJSBufferSource = class external name ''BufferSource''',
  18182. ' end;',
  18183. 'procedure DoIt(s: TJSBufferSource); external name ''DoIt'';',
  18184. '']),
  18185. '');
  18186. AddModuleWithIntfImplSrc('unit3.pas',
  18187. LinesToStr([
  18188. '{$modeswitch externalclass}',
  18189. 'type',
  18190. ' TJSBufferSource = class external name ''BufferSource''',
  18191. ' end;',
  18192. '']),
  18193. '');
  18194. StartUnit(true);
  18195. Add([
  18196. 'interface',
  18197. 'uses unit2, unit3;',
  18198. 'procedure DoSome(s: TJSBufferSource);',
  18199. 'implementation',
  18200. 'procedure DoSome(s: TJSBufferSource);',
  18201. 'begin',
  18202. ' DoIt(s);',
  18203. 'end;',
  18204. '']);
  18205. SetExpectedPasResolverError('Incompatible type arg no. 1: Got "unit3.TJSBufferSource", expected "unit2.TJSBufferSource"',
  18206. nIncompatibleTypeArgNo);
  18207. ConvertUnit;
  18208. end;
  18209. procedure TTestModule.TestClassInterface_Corba;
  18210. begin
  18211. StartProgram(false);
  18212. Add([
  18213. '{$interfaces corba}',
  18214. 'type',
  18215. ' IUnknown = interface;',
  18216. ' IUnknown = interface',
  18217. ' [''{00000000-0000-0000-C000-000000000046}'']',
  18218. ' end;',
  18219. ' IInterface = IUnknown;',
  18220. ' IBird = interface(IInterface)',
  18221. ' function GetSize: longint;',
  18222. ' procedure SetSize(i: longint);',
  18223. ' property Size: longint read GetSize write SetSize;',
  18224. ' procedure DoIt(i: longint);',
  18225. ' end;',
  18226. ' TObject = class',
  18227. ' end;',
  18228. ' TBird = class(TObject,IBird)',
  18229. ' function GetSize: longint; virtual; abstract;',
  18230. ' procedure SetSize(i: longint); virtual; abstract;',
  18231. ' procedure DoIt(i: longint); virtual; abstract;',
  18232. ' end;',
  18233. 'var',
  18234. ' BirdIntf: IBird;',
  18235. 'begin',
  18236. ' BirdIntf.Size:=BirdIntf.Size;',
  18237. '']);
  18238. ConvertProgram;
  18239. CheckSource('TestClassInterface_Corba',
  18240. LinesToStr([ // statements
  18241. 'rtl.createInterface(this, "IUnknown", "{00000000-0000-0000-C000-000000000046}", [], null);',
  18242. 'rtl.createInterface(this, "IBird", "{5BD1A53B-69BB-37EE-AF32-BEFB86D85B03}", ["GetSize", "SetSize", "DoIt"], this.IUnknown);',
  18243. 'rtl.createClass(this, "TObject", null, function () {',
  18244. ' this.$init = function () {',
  18245. ' };',
  18246. ' this.$final = function () {',
  18247. ' };',
  18248. '});',
  18249. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18250. ' rtl.addIntf(this, $mod.IBird);',
  18251. '});',
  18252. 'this.BirdIntf = null;',
  18253. '']),
  18254. LinesToStr([ // $mod.$main
  18255. ' $mod.BirdIntf.SetSize($mod.BirdIntf.GetSize());',
  18256. '']));
  18257. end;
  18258. procedure TTestModule.TestClassInterface_ProcExternalFail;
  18259. begin
  18260. StartProgram(false);
  18261. Add([
  18262. '{$interfaces corba}',
  18263. 'type',
  18264. ' IUnknown = interface',
  18265. ' procedure DoIt; external name ''foo'';',
  18266. ' end;',
  18267. 'begin']);
  18268. SetExpectedParserError(
  18269. 'Fields are not allowed in interface at token "Identifier external" in file test1.pp at line 6 column 21',
  18270. nParserNoFieldsAllowed);
  18271. ConvertProgram;
  18272. end;
  18273. procedure TTestModule.TestClassInterface_Overloads;
  18274. begin
  18275. StartProgram(false);
  18276. Add([
  18277. '{$interfaces corba}',
  18278. 'type',
  18279. ' integer = longint;',
  18280. ' IUnknown = interface',
  18281. ' procedure DoIt(i: integer);',
  18282. ' procedure DoIt(s: string);',
  18283. ' end;',
  18284. ' IBird = interface(IUnknown)',
  18285. ' procedure DoIt(b: boolean); overload;',
  18286. ' end;',
  18287. ' TObject = class',
  18288. ' end;',
  18289. ' TBird = class(TObject,IBird)',
  18290. ' procedure DoIt(o: TObject);',
  18291. ' procedure DoIt(s: string);',
  18292. ' procedure DoIt(i: integer);',
  18293. ' procedure DoIt(b: boolean);',
  18294. ' end;',
  18295. 'procedure TBird.DoIt(o: TObject); begin end;',
  18296. 'procedure TBird.DoIt(s: string); begin end;',
  18297. 'procedure TBird.DoIt(i: integer); begin end;',
  18298. 'procedure TBird.DoIt(b: boolean); begin end;',
  18299. 'var',
  18300. ' BirdIntf: IBird;',
  18301. 'begin',
  18302. ' BirdIntf.DoIt(3);',
  18303. ' BirdIntf.DoIt(''abc'');',
  18304. ' BirdIntf.DoIt(true);',
  18305. '']);
  18306. ConvertProgram;
  18307. CheckSource('TestClassInterface_Overloads',
  18308. LinesToStr([ // statements
  18309. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-BDC4-8A2AE2C59400}", ["DoIt", "DoIt$1"], null);',
  18310. 'rtl.createInterface(this, "IBird", "{8285DD5E-EA3E-396E-AE88-000B86AABF05}", ["DoIt$2"], this.IUnknown);',
  18311. 'rtl.createClass(this, "TObject", null, function () {',
  18312. ' this.$init = function () {',
  18313. ' };',
  18314. ' this.$final = function () {',
  18315. ' };',
  18316. '});',
  18317. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18318. ' this.DoIt = function (o) {',
  18319. ' };',
  18320. ' this.DoIt$1 = function (s) {',
  18321. ' };',
  18322. ' this.DoIt$2 = function (i) {',
  18323. ' };',
  18324. ' this.DoIt$3 = function (b) {',
  18325. ' };',
  18326. ' rtl.addIntf(this, $mod.IBird, {',
  18327. ' DoIt$2: "DoIt$3",',
  18328. ' DoIt: "DoIt$2"',
  18329. ' });',
  18330. '});',
  18331. 'this.BirdIntf = null;',
  18332. '']),
  18333. LinesToStr([ // $mod.$main
  18334. '$mod.BirdIntf.DoIt(3);',
  18335. '$mod.BirdIntf.DoIt$1("abc");',
  18336. '$mod.BirdIntf.DoIt$2(true);',
  18337. '']));
  18338. end;
  18339. procedure TTestModule.TestClassInterface_DuplicateGUIInIntfListFail;
  18340. begin
  18341. StartProgram(false);
  18342. Add([
  18343. '{$interfaces corba}',
  18344. 'type',
  18345. ' IBird = interface',
  18346. ' [''{4B3BA825-E0EC-4799-A19C-55F714A07959}'']',
  18347. ' end;',
  18348. ' IDog = interface',
  18349. ' [''{4B3BA825-E0EC-4799-A19C-55F714A07959}'']',
  18350. ' end;',
  18351. ' TObject = class(IBird,IDog)',
  18352. ' end;',
  18353. 'begin']);
  18354. SetExpectedPasResolverError('Duplicate GUID {4B3BA825-E0EC-4799-A19C-55F714A07959} in IDog and IBird',
  18355. nDuplicateGUIDXInYZ);
  18356. ConvertProgram;
  18357. end;
  18358. procedure TTestModule.TestClassInterface_DuplicateGUIInAncestorFail;
  18359. begin
  18360. StartProgram(false);
  18361. Add([
  18362. '{$interfaces corba}',
  18363. 'type',
  18364. ' IAnimal = interface',
  18365. ' [''{4B3BA825-E0EC-4799-A19C-55F714A07959}'']',
  18366. ' end;',
  18367. ' IBird = interface(IAnimal)',
  18368. ' end;',
  18369. ' IHawk = interface(IBird)',
  18370. ' [''{4B3BA825-E0EC-4799-A19C-55F714A07959}'']',
  18371. ' end;',
  18372. 'begin']);
  18373. SetExpectedPasResolverError('Duplicate GUID {4B3BA825-E0EC-4799-A19C-55F714A07959} in IHawk and IAnimal',
  18374. nDuplicateGUIDXInYZ);
  18375. ConvertProgram;
  18376. end;
  18377. procedure TTestModule.TestClassInterface_AncestorImpl;
  18378. begin
  18379. StartProgram(false);
  18380. Add([
  18381. '{$interfaces corba}',
  18382. 'type',
  18383. ' integer = longint;',
  18384. ' IUnknown = interface',
  18385. ' procedure DoIt(i: integer);',
  18386. ' end;',
  18387. ' IBird = interface',
  18388. ' procedure Fly(i: integer);',
  18389. ' end;',
  18390. ' TObject = class(IUnknown)',
  18391. ' procedure DoIt(i: integer);',
  18392. ' end;',
  18393. ' TBird = class(IBird)',
  18394. ' procedure Fly(i: integer);',
  18395. ' end;',
  18396. 'procedure TObject.DoIt(i: integer); begin end;',
  18397. 'procedure TBird.Fly(i: integer); begin end;',
  18398. 'begin',
  18399. '']);
  18400. ConvertProgram;
  18401. CheckSource('TestClassInterface_AncestorIntf',
  18402. LinesToStr([ // statements
  18403. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-BDC4-8A2800000000}", ["DoIt"], null);',
  18404. 'rtl.createInterface(this, "IBird", "{B92D5841-6264-3AE3-BF20-000000000000}", ["Fly"], null);',
  18405. 'rtl.createClass(this, "TObject", null, function () {',
  18406. ' this.$init = function () {',
  18407. ' };',
  18408. ' this.$final = function () {',
  18409. ' };',
  18410. ' this.DoIt = function (i) {',
  18411. ' };',
  18412. ' rtl.addIntf(this, $mod.IUnknown);',
  18413. '});',
  18414. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18415. ' this.Fly = function (i) {',
  18416. ' };',
  18417. ' rtl.addIntf(this, $mod.IBird);',
  18418. ' rtl.addIntf(this, $mod.IUnknown);',
  18419. '});',
  18420. '']),
  18421. LinesToStr([ // $mod.$main
  18422. '']));
  18423. end;
  18424. procedure TTestModule.TestClassInterface_ImplReintroduce;
  18425. begin
  18426. StartProgram(false);
  18427. Add([
  18428. '{$interfaces corba}',
  18429. 'type',
  18430. ' integer = longint;',
  18431. ' IBird = interface',
  18432. ' procedure DoIt(i: integer);',
  18433. ' end;',
  18434. ' TObject = class',
  18435. ' procedure DoIt(i: integer);',
  18436. ' end;',
  18437. ' TBird = class(IBird)',
  18438. ' procedure DoIt(i: integer); virtual; reintroduce;',
  18439. ' end;',
  18440. 'procedure TObject.DoIt(i: integer); begin end;',
  18441. 'procedure TBird.DoIt(i: integer); begin end;',
  18442. 'begin',
  18443. '']);
  18444. ConvertProgram;
  18445. CheckSource('TestClassInterface_ImplReintroduce',
  18446. LinesToStr([ // statements
  18447. 'rtl.createInterface(this, "IBird", "{B92D5841-6264-3AE2-8594-000000000000}", ["DoIt"], null);',
  18448. 'rtl.createClass(this, "TObject", null, function () {',
  18449. ' this.$init = function () {',
  18450. ' };',
  18451. ' this.$final = function () {',
  18452. ' };',
  18453. ' this.DoIt = function (i) {',
  18454. ' };',
  18455. '});',
  18456. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18457. ' this.DoIt$1 = function (i) {',
  18458. ' };',
  18459. ' rtl.addIntf(this, $mod.IBird, {',
  18460. ' DoIt: "DoIt$1"',
  18461. ' });',
  18462. '});',
  18463. '']),
  18464. LinesToStr([ // $mod.$main
  18465. '']));
  18466. end;
  18467. procedure TTestModule.TestClassInterface_MethodResolution;
  18468. begin
  18469. StartProgram(false);
  18470. Add([
  18471. '{$interfaces corba}',
  18472. 'type',
  18473. ' IUnknown = interface',
  18474. ' procedure Walk(i: longint);',
  18475. ' end;',
  18476. ' IBird = interface(IUnknown)',
  18477. ' procedure Walk(b: boolean); overload;',
  18478. ' procedure Fly(s: string);',
  18479. ' end;',
  18480. ' TObject = class',
  18481. ' end;',
  18482. ' TBird = class(TObject,IBird)',
  18483. ' procedure IBird.Fly = Move;',
  18484. ' procedure IBird.Walk = Hop;',
  18485. ' procedure Hop(i: longint);',
  18486. ' procedure Move(s: string);',
  18487. ' procedure Hop(b: boolean);',
  18488. ' end;',
  18489. 'procedure TBird.Move(s: string); begin end;',
  18490. 'procedure TBird.Hop(i: longint); begin end;',
  18491. 'procedure TBird.Hop(b: boolean); begin end;',
  18492. 'var',
  18493. ' BirdIntf: IBird;',
  18494. 'begin',
  18495. ' BirdIntf.Walk(3);',
  18496. ' BirdIntf.Walk(true);',
  18497. ' BirdIntf.Fly(''abc'');',
  18498. '']);
  18499. ConvertProgram;
  18500. CheckSource('TestClassInterface_MethodResolution',
  18501. LinesToStr([ // statements
  18502. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-BDD7-23D600000000}", ["Walk"], null);',
  18503. 'rtl.createInterface(this, "IBird", "{CF8A4986-80F6-396E-AE88-000B86AAE208}", ["Walk$1", "Fly"], this.IUnknown);',
  18504. 'rtl.createClass(this, "TObject", null, function () {',
  18505. ' this.$init = function () {',
  18506. ' };',
  18507. ' this.$final = function () {',
  18508. ' };',
  18509. '});',
  18510. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18511. ' this.Hop = function (i) {',
  18512. ' };',
  18513. ' this.Move = function (s) {',
  18514. ' };',
  18515. ' this.Hop$1 = function (b) {',
  18516. ' };',
  18517. ' rtl.addIntf(this, $mod.IBird, {',
  18518. ' Walk$1: "Hop$1",',
  18519. ' Fly: "Move",',
  18520. ' Walk: "Hop"',
  18521. ' });',
  18522. '});',
  18523. 'this.BirdIntf = null;',
  18524. '']),
  18525. LinesToStr([ // $mod.$main
  18526. '$mod.BirdIntf.Walk(3);',
  18527. '$mod.BirdIntf.Walk$1(true);',
  18528. '$mod.BirdIntf.Fly("abc");',
  18529. '']));
  18530. end;
  18531. procedure TTestModule.TestClassInterface_AncestorMoreInterfaces;
  18532. begin
  18533. StartProgram(false);
  18534. Add([
  18535. '{$interfaces com}',
  18536. 'type',
  18537. ' IUnknown = interface',
  18538. ' function _AddRef: longint;',
  18539. ' procedure Walk;',
  18540. ' end;',
  18541. ' IBird = interface end;',
  18542. ' IDog = interface end;',
  18543. ' TObject = class(IBird,IDog)',
  18544. ' function _AddRef: longint; virtual; abstract;',
  18545. ' procedure Walk; virtual; abstract;',
  18546. ' end;',
  18547. ' TBird = class(IUnknown)',
  18548. ' end;',
  18549. 'begin',
  18550. '']);
  18551. ConvertProgram;
  18552. CheckSource('TestClassInterface_COM_AncestorLess',
  18553. LinesToStr([ // statements
  18554. 'rtl.createInterface(this, "IUnknown", "{8F2D5841-758A-322B-BDDF-21CD521DD723}", ["_AddRef", "Walk"], null);',
  18555. 'rtl.createInterface(this, "IBird", "{CCE11D4C-6504-3AEE-AE88-000B86AAE675}", [], this.IUnknown);',
  18556. 'rtl.createInterface(this, "IDog", "{CCE11D4C-6504-3AEE-AE88-000B8E5FC675}", [], this.IUnknown);',
  18557. 'rtl.createClass(this, "TObject", null, function () {',
  18558. ' this.$init = function () {',
  18559. ' };',
  18560. ' this.$final = function () {',
  18561. ' };',
  18562. ' rtl.addIntf(this, $mod.IBird);',
  18563. ' rtl.addIntf(this, $mod.IDog);',
  18564. '});',
  18565. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18566. ' rtl.addIntf(this, $mod.IUnknown);',
  18567. ' rtl.addIntf(this, $mod.IBird);',
  18568. ' rtl.addIntf(this, $mod.IDog);',
  18569. '});',
  18570. '']),
  18571. LinesToStr([ // $mod.$main
  18572. '']));
  18573. end;
  18574. procedure TTestModule.TestClassInterface_MethodOverride;
  18575. begin
  18576. StartProgram(false);
  18577. Add([
  18578. '{$interfaces corba}',
  18579. 'type',
  18580. ' IUnknown = interface',
  18581. ' [''{D6D98E5B-8A10-4FEC-856A-7BFC847FE74B}'']',
  18582. ' procedure Go;',
  18583. ' end;',
  18584. ' TObject = class(IUnknown)',
  18585. ' procedure Go; virtual; abstract;',
  18586. ' end;',
  18587. ' TBird = class',
  18588. ' procedure Go; override;',
  18589. ' end;',
  18590. ' TCat = class(TObject)',
  18591. ' procedure Go; override;',
  18592. ' end;',
  18593. ' TDog = class(TObject, IUnknown)',
  18594. ' procedure Go; override;',
  18595. ' end;',
  18596. 'procedure TBird.Go; begin end;',
  18597. 'procedure TCat.Go; begin end;',
  18598. 'procedure TDog.Go; begin end;',
  18599. 'begin',
  18600. '']);
  18601. ConvertProgram;
  18602. CheckSource('TestClassInterface_MethodOverride',
  18603. LinesToStr([ // statements
  18604. 'rtl.createInterface(this, "IUnknown", "{D6D98E5B-8A10-4FEC-856A-7BFC847FE74B}", ["Go"], null);',
  18605. 'rtl.createClass(this, "TObject", null, function () {',
  18606. ' this.$init = function () {',
  18607. ' };',
  18608. ' this.$final = function () {',
  18609. ' };',
  18610. ' rtl.addIntf(this, $mod.IUnknown);',
  18611. '});',
  18612. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18613. ' this.Go = function () {',
  18614. ' };',
  18615. ' rtl.addIntf(this, $mod.IUnknown);',
  18616. '});',
  18617. 'rtl.createClass(this, "TCat", this.TObject, function () {',
  18618. ' this.Go = function () {',
  18619. ' };',
  18620. ' rtl.addIntf(this, $mod.IUnknown);',
  18621. '});',
  18622. 'rtl.createClass(this, "TDog", this.TObject, function () {',
  18623. ' this.Go = function () {',
  18624. ' };',
  18625. ' rtl.addIntf(this, $mod.IUnknown);',
  18626. '});',
  18627. '']),
  18628. LinesToStr([ // $mod.$main
  18629. '']));
  18630. end;
  18631. procedure TTestModule.TestClassInterface_Corba_Delegation;
  18632. begin
  18633. StartProgram(false);
  18634. Add([
  18635. '{$interfaces corba}',
  18636. 'type',
  18637. ' IUnknown = interface',
  18638. ' end;',
  18639. ' IBird = interface(IUnknown)',
  18640. ' procedure Fly(s: string);',
  18641. ' end;',
  18642. ' IEagle = interface(IBird)',
  18643. ' end;',
  18644. ' IDove = interface(IBird)',
  18645. ' end;',
  18646. ' ISwallow = interface(IBird)',
  18647. ' end;',
  18648. ' TObject = class',
  18649. ' end;',
  18650. ' TBird = class(TObject,IBird,IEagle,IDove,ISwallow)',
  18651. ' procedure Fly(s: string); virtual; abstract;',
  18652. ' end;',
  18653. ' TBat = class(IBird,IEagle,IDove,ISwallow)',
  18654. ' FBirdIntf: IBird;',
  18655. ' property BirdIntf: IBird read FBirdIntf implements IBird;',
  18656. ' function GetEagleIntf: IEagle; virtual; abstract;',
  18657. ' property EagleIntf: IEagle read GetEagleIntf implements IEagle;',
  18658. ' FDoveObj: TBird;',
  18659. ' property DoveObj: TBird read FDoveObj implements IDove;',
  18660. ' function GetSwallowObj: TBird; virtual; abstract;',
  18661. ' property SwallowObj: TBird read GetSwallowObj implements ISwallow;',
  18662. ' end;',
  18663. 'begin',
  18664. '']);
  18665. ConvertProgram;
  18666. CheckSource('TestClassInterface_Corba_Delegation',
  18667. LinesToStr([ // statements
  18668. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  18669. 'rtl.createInterface(this, "IBird", "{478D080B-C0F6-396E-AE88-000B87785B07}", ["Fly"], this.IUnknown);',
  18670. 'rtl.createInterface(this, "IEagle", "{489289DE-FDE2-34A6-8288-39119022B1B4}", [], this.IBird);',
  18671. 'rtl.createInterface(this, "IDove", "{489289DE-FDE2-34A6-8288-39118EF16074}", [], this.IBird);',
  18672. 'rtl.createInterface(this, "ISwallow", "{B89289DE-FDE2-34A6-8288-3911CBDCB359}", [], this.IBird);',
  18673. 'rtl.createClass(this, "TObject", null, function () {',
  18674. ' this.$init = function () {',
  18675. ' };',
  18676. ' this.$final = function () {',
  18677. ' };',
  18678. '});',
  18679. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18680. ' rtl.addIntf(this, $mod.IBird);',
  18681. ' rtl.addIntf(this, $mod.IEagle);',
  18682. ' rtl.addIntf(this, $mod.IDove);',
  18683. ' rtl.addIntf(this, $mod.ISwallow);',
  18684. '});',
  18685. 'rtl.createClass(this, "TBat", this.TObject, function () {',
  18686. ' this.$init = function () {',
  18687. ' $mod.TObject.$init.call(this);',
  18688. ' this.FBirdIntf = null;',
  18689. ' this.FDoveObj = null;',
  18690. ' };',
  18691. ' this.$final = function () {',
  18692. ' this.FBirdIntf = undefined;',
  18693. ' this.FDoveObj = undefined;',
  18694. ' $mod.TObject.$final.call(this);',
  18695. ' };',
  18696. ' this.$intfmaps = {',
  18697. ' "{478D080B-C0F6-396E-AE88-000B87785B07}": function () {',
  18698. ' return this.FBirdIntf;',
  18699. ' },',
  18700. ' "{489289DE-FDE2-34A6-8288-39119022B1B4}": function () {',
  18701. ' return this.GetEagleIntf();',
  18702. ' },',
  18703. ' "{489289DE-FDE2-34A6-8288-39118EF16074}": function () {',
  18704. ' return rtl.getIntfT(this.FDoveObj, $mod.IDove);',
  18705. ' },',
  18706. ' "{B89289DE-FDE2-34A6-8288-3911CBDCB359}": function () {',
  18707. ' return rtl.getIntfT(this.GetSwallowObj(), $mod.ISwallow);',
  18708. ' }',
  18709. ' };',
  18710. '});',
  18711. '']),
  18712. LinesToStr([ // $mod.$main
  18713. '']));
  18714. end;
  18715. procedure TTestModule.TestClassInterface_Corba_DelegationStatic;
  18716. begin
  18717. StartProgram(false);
  18718. Add([
  18719. '{$interfaces corba}',
  18720. 'type',
  18721. ' IUnknown = interface',
  18722. ' end;',
  18723. ' IBird = interface(IUnknown)',
  18724. ' procedure Fly(s: string);',
  18725. ' end;',
  18726. ' IEagle = interface(IBird)',
  18727. ' end;',
  18728. ' IDove = interface(IBird)',
  18729. ' end;',
  18730. ' ISwallow = interface(IBird)',
  18731. ' end;',
  18732. ' TObject = class',
  18733. ' end;',
  18734. ' TBird = class(TObject,IBird,IEagle,IDove,ISwallow)',
  18735. ' procedure Fly(s: string); virtual; abstract;',
  18736. ' end;',
  18737. ' TBat = class(IBird,IEagle,IDove,ISwallow)',
  18738. ' private',
  18739. ' class var FBirdIntf: IBird;',
  18740. ' class var FDoveObj: TBird;',
  18741. ' class function GetEagleIntf: IEagle; virtual; abstract;',
  18742. ' class function GetSwallowObj: TBird; virtual; abstract;',
  18743. ' protected',
  18744. ' class property BirdIntf: IBird read FBirdIntf implements IBird;',
  18745. ' class property EagleIntf: IEagle read GetEagleIntf implements IEagle;',
  18746. ' class property DoveObj: TBird read FDoveObj implements IDove;',
  18747. ' class property SwallowObj: TBird read GetSwallowObj implements ISwallow;',
  18748. ' end;',
  18749. 'begin',
  18750. '']);
  18751. ConvertProgram;
  18752. CheckSource('TestClassInterface_Corba_DelegationStatic',
  18753. LinesToStr([ // statements
  18754. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  18755. 'rtl.createInterface(this, "IBird", "{478D080B-C0F6-396E-AE88-000B87785B07}", ["Fly"], this.IUnknown);',
  18756. 'rtl.createInterface(this, "IEagle", "{489289DE-FDE2-34A6-8288-39119022B1B4}", [], this.IBird);',
  18757. 'rtl.createInterface(this, "IDove", "{489289DE-FDE2-34A6-8288-39118EF16074}", [], this.IBird);',
  18758. 'rtl.createInterface(this, "ISwallow", "{B89289DE-FDE2-34A6-8288-3911CBDCB359}", [], this.IBird);',
  18759. 'rtl.createClass(this, "TObject", null, function () {',
  18760. ' this.$init = function () {',
  18761. ' };',
  18762. ' this.$final = function () {',
  18763. ' };',
  18764. '});',
  18765. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18766. ' rtl.addIntf(this, $mod.IBird);',
  18767. ' rtl.addIntf(this, $mod.IEagle);',
  18768. ' rtl.addIntf(this, $mod.IDove);',
  18769. ' rtl.addIntf(this, $mod.ISwallow);',
  18770. '});',
  18771. 'rtl.createClass(this, "TBat", this.TObject, function () {',
  18772. ' this.FBirdIntf = null;',
  18773. ' this.FDoveObj = null;',
  18774. ' this.$intfmaps = {',
  18775. ' "{478D080B-C0F6-396E-AE88-000B87785B07}": function () {',
  18776. ' return this.FBirdIntf;',
  18777. ' },',
  18778. ' "{489289DE-FDE2-34A6-8288-39119022B1B4}": function () {',
  18779. ' return this.GetEagleIntf();',
  18780. ' },',
  18781. ' "{489289DE-FDE2-34A6-8288-39118EF16074}": function () {',
  18782. ' return rtl.getIntfT(this.FDoveObj, $mod.IDove);',
  18783. ' },',
  18784. ' "{B89289DE-FDE2-34A6-8288-3911CBDCB359}": function () {',
  18785. ' return rtl.getIntfT(this.GetSwallowObj(), $mod.ISwallow);',
  18786. ' }',
  18787. ' };',
  18788. '});',
  18789. '']),
  18790. LinesToStr([ // $mod.$main
  18791. '']));
  18792. end;
  18793. procedure TTestModule.TestClassInterface_Corba_Operators;
  18794. begin
  18795. StartProgram(false);
  18796. Add([
  18797. '{$interfaces corba}',
  18798. 'type',
  18799. ' IUnknown = interface',
  18800. ' end;',
  18801. ' IBird = interface(IUnknown)',
  18802. ' function GetItems(Index: longint): longint;',
  18803. ' procedure SetItems(Index: longint; Value: longint);',
  18804. ' property Items[Index: longint]: longint read GetItems write SetItems; default;',
  18805. ' end;',
  18806. ' TObject = class',
  18807. ' end;',
  18808. ' TBird = class(TObject,IBird)',
  18809. ' function GetItems(Index: longint): longint; virtual; abstract;',
  18810. ' procedure SetItems(Index: longint; Value: longint); virtual; abstract;',
  18811. ' end;',
  18812. 'var',
  18813. ' IntfVar: IBird = nil;',
  18814. ' IntfVar2: IBird;',
  18815. ' ObjVar: TBird;',
  18816. ' v: JSValue;',
  18817. 'begin',
  18818. ' IntfVar:=nil;',
  18819. ' IntfVar[3]:=IntfVar[4];',
  18820. ' if Assigned(IntfVar) then ;',
  18821. ' IntfVar:=IntfVar2;',
  18822. ' IntfVar:=ObjVar;',
  18823. ' if IntfVar=IntfVar2 then ;',
  18824. ' if IntfVar<>IntfVar2 then ;',
  18825. ' if IntfVar is IBird then ;',
  18826. ' if IntfVar is TBird then ;',
  18827. ' if ObjVar is IBird then ;',
  18828. ' IntfVar:=IntfVar2 as IBird;',
  18829. ' ObjVar:=IntfVar2 as TBird;',
  18830. ' IntfVar:=ObjVar as IBird;',
  18831. ' IntfVar:=IBird(IntfVar2);',
  18832. ' ObjVar:=TBird(IntfVar);',
  18833. ' IntfVar:=IBird(ObjVar);',
  18834. ' v:=IntfVar;',
  18835. ' IntfVar:=IBird(v);',
  18836. ' if v is IBird then ;',
  18837. ' v:=JSValue(IntfVar);',
  18838. ' v:=IBird;',
  18839. '']);
  18840. ConvertProgram;
  18841. CheckSource('TestClassInterface_Corba_Operators',
  18842. LinesToStr([ // statements
  18843. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  18844. 'rtl.createInterface(this, "IBird", "{D53FED90-DE59-3202-B1AE-000B87785B08}", ["GetItems", "SetItems"], this.IUnknown);',
  18845. 'rtl.createClass(this, "TObject", null, function () {',
  18846. ' this.$init = function () {',
  18847. ' };',
  18848. ' this.$final = function () {',
  18849. ' };',
  18850. '});',
  18851. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18852. ' rtl.addIntf(this, $mod.IBird);',
  18853. '});',
  18854. 'this.IntfVar = null;',
  18855. 'this.IntfVar2 = null;',
  18856. 'this.ObjVar = null;',
  18857. 'this.v = undefined;',
  18858. '']),
  18859. LinesToStr([ // $mod.$main
  18860. '$mod.IntfVar = null;',
  18861. '$mod.IntfVar.SetItems(3, $mod.IntfVar.GetItems(4));',
  18862. 'if ($mod.IntfVar != null) ;',
  18863. '$mod.IntfVar = $mod.IntfVar2;',
  18864. '$mod.IntfVar = rtl.getIntfT($mod.ObjVar,$mod.IBird);',
  18865. 'if ($mod.IntfVar === $mod.IntfVar2) ;',
  18866. 'if ($mod.IntfVar !== $mod.IntfVar2) ;',
  18867. 'if ($mod.IBird.isPrototypeOf($mod.IntfVar)) ;',
  18868. 'if (rtl.intfIsClass($mod.IntfVar, $mod.TBird)) ;',
  18869. 'if (rtl.getIntfT($mod.ObjVar, $mod.IBird) !== null) ;',
  18870. '$mod.IntfVar = rtl.as($mod.IntfVar2, $mod.IBird);',
  18871. '$mod.ObjVar = rtl.intfAsClass($mod.IntfVar2, $mod.TBird);',
  18872. '$mod.IntfVar = rtl.getIntfT($mod.ObjVar, $mod.IBird);',
  18873. '$mod.IntfVar = $mod.IntfVar2;',
  18874. '$mod.ObjVar = rtl.intfToClass($mod.IntfVar, $mod.TBird);',
  18875. '$mod.IntfVar = rtl.getIntfT($mod.ObjVar, $mod.IBird);',
  18876. '$mod.v = $mod.IntfVar;',
  18877. '$mod.IntfVar = rtl.getObject($mod.v);',
  18878. 'if (rtl.isExt($mod.v, $mod.IBird, 1)) ;',
  18879. '$mod.v = $mod.IntfVar;',
  18880. '$mod.v = $mod.IBird;',
  18881. '']));
  18882. end;
  18883. procedure TTestModule.TestClassInterface_Corba_Args;
  18884. begin
  18885. StartProgram(false);
  18886. Add([
  18887. '{$interfaces corba}',
  18888. 'type',
  18889. ' IUnknown = interface',
  18890. ' end;',
  18891. ' IBird = interface(IUnknown)',
  18892. ' end;',
  18893. ' TObject = class',
  18894. ' end;',
  18895. ' TBird = class(TObject,IBird)',
  18896. ' end;',
  18897. 'procedure DoIt(var u; i: IBird; const j: IBird);',
  18898. 'begin',
  18899. ' DoIt(i,i,i);',
  18900. 'end;',
  18901. 'procedure Change(var i: IBird; out j: IBird);',
  18902. 'begin',
  18903. ' DoIt(i,i,i);',
  18904. ' Change(i,i);',
  18905. 'end;',
  18906. 'var',
  18907. ' i: IBird;',
  18908. ' o: TBird;',
  18909. 'begin',
  18910. ' DoIt(i,i,i);',
  18911. ' Change(i,i);',
  18912. ' DoIt(o,o,o);',
  18913. '']);
  18914. ConvertProgram;
  18915. CheckSource('TestClassInterface_Corba_Args',
  18916. LinesToStr([ // statements
  18917. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  18918. 'rtl.createInterface(this, "IBird", "{4B0D080B-C0F6-396E-AE88-000B87785074}", [], this.IUnknown);',
  18919. 'rtl.createClass(this, "TObject", null, function () {',
  18920. ' this.$init = function () {',
  18921. ' };',
  18922. ' this.$final = function () {',
  18923. ' };',
  18924. '});',
  18925. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  18926. ' rtl.addIntf(this, $mod.IBird);',
  18927. '});',
  18928. 'this.DoIt = function (u, i, j) {',
  18929. ' $mod.DoIt({',
  18930. ' get: function () {',
  18931. ' return i;',
  18932. ' },',
  18933. ' set: function (v) {',
  18934. ' i = v;',
  18935. ' }',
  18936. ' }, i, i);',
  18937. '};',
  18938. 'this.Change = function (i, j) {',
  18939. ' $mod.DoIt(i, i.get(), i.get());',
  18940. ' $mod.Change(i, i);',
  18941. '};',
  18942. 'this.i = null;',
  18943. 'this.o = null;',
  18944. '']),
  18945. LinesToStr([ // $mod.$main
  18946. '$mod.DoIt({',
  18947. ' p: $mod,',
  18948. ' get: function () {',
  18949. ' return this.p.i;',
  18950. ' },',
  18951. ' set: function (v) {',
  18952. ' this.p.i = v;',
  18953. ' }',
  18954. '}, $mod.i, $mod.i);',
  18955. '$mod.Change({',
  18956. ' p: $mod,',
  18957. ' get: function () {',
  18958. ' return this.p.i;',
  18959. ' },',
  18960. ' set: function (v) {',
  18961. ' this.p.i = v;',
  18962. ' }',
  18963. '}, {',
  18964. ' p: $mod,',
  18965. ' get: function () {',
  18966. ' return this.p.i;',
  18967. ' },',
  18968. ' set: function (v) {',
  18969. ' this.p.i = v;',
  18970. ' }',
  18971. '});',
  18972. '$mod.DoIt({',
  18973. ' p: $mod,',
  18974. ' get: function () {',
  18975. ' return this.p.o;',
  18976. ' },',
  18977. ' set: function (v) {',
  18978. ' this.p.o = v;',
  18979. ' }',
  18980. '}, rtl.getIntfT($mod.o, $mod.IBird), rtl.getIntfT($mod.o, $mod.IBird));',
  18981. '']));
  18982. end;
  18983. procedure TTestModule.TestClassInterface_Corba_ForIn;
  18984. begin
  18985. StartProgram(false);
  18986. Add([
  18987. '{$interfaces corba}',
  18988. 'type',
  18989. ' IUnknown = interface end;',
  18990. ' TObject = class',
  18991. ' Id: longint;',
  18992. ' end;',
  18993. ' IEnumerator = interface(IUnknown)',
  18994. ' function GetCurrent: TObject;',
  18995. ' function MoveNext: Boolean;',
  18996. ' property Current: TObject read GetCurrent;',
  18997. ' end;',
  18998. ' IEnumerable = interface(IUnknown)',
  18999. ' function GetEnumerator: IEnumerator;',
  19000. ' end;',
  19001. 'var',
  19002. ' o: TObject;',
  19003. ' i: IEnumerable;',
  19004. 'begin',
  19005. ' for o in i do o.Id:=3;',
  19006. '']);
  19007. ConvertProgram;
  19008. CheckSource('TestClassInterface_Corba_ForIn',
  19009. LinesToStr([ // statements
  19010. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  19011. 'rtl.createClass(this, "TObject", null, function () {',
  19012. ' this.$init = function () {',
  19013. ' this.Id = 0;',
  19014. ' };',
  19015. ' this.$final = function () {',
  19016. ' };',
  19017. '});',
  19018. 'rtl.createInterface(this, "IEnumerator", "{95D7745D-ED61-3F13-BBE4-07708161999E}", ["GetCurrent", "MoveNext"], this.IUnknown);',
  19019. 'rtl.createInterface(this, "IEnumerable", "{8CC9D45D-ED7D-3B73-96B6-290B931BB19E}", ["GetEnumerator"], this.IUnknown);',
  19020. 'this.o = null;',
  19021. 'this.i = null;',
  19022. '']),
  19023. LinesToStr([ // $mod.$main
  19024. 'var $in = $mod.i.GetEnumerator();',
  19025. 'while ($in.MoveNext()) {',
  19026. ' $mod.o = $in.GetCurrent();',
  19027. ' $mod.o.Id = 3;',
  19028. '};',
  19029. '']));
  19030. end;
  19031. procedure TTestModule.TestClassInterface_COM_AssignVar;
  19032. begin
  19033. StartProgram(false);
  19034. Add([
  19035. '{$interfaces com}',
  19036. 'type',
  19037. ' IUnknown = interface',
  19038. ' function _AddRef: longint;',
  19039. ' function _Release: longint;',
  19040. ' end;',
  19041. ' TObject = class(IUnknown)',
  19042. ' function _AddRef: longint; virtual; abstract;',
  19043. ' function _Release: longint; virtual; abstract;',
  19044. ' end;',
  19045. 'var',
  19046. ' i: IUnknown;',
  19047. 'procedure DoGlobal(o: TObject);',
  19048. 'begin',
  19049. ' i:=nil;',
  19050. ' i:=o;',
  19051. ' i:=i;',
  19052. 'end;',
  19053. 'procedure DoLocal(o: TObject);',
  19054. 'const k: IUnknown = nil;',
  19055. 'var j: IUnknown;',
  19056. 'begin',
  19057. ' k:=o;',
  19058. ' k:=i;',
  19059. ' j:=o;',
  19060. ' j:=i;',
  19061. 'end;',
  19062. 'var o: TObject;',
  19063. 'begin',
  19064. ' i:=nil;',
  19065. ' i:=o;',
  19066. '']);
  19067. ConvertProgram;
  19068. CheckSource('TestClassInterface_COM_AssignVar',
  19069. LinesToStr([ // statements
  19070. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19071. 'rtl.createClass(this, "TObject", null, function () {',
  19072. ' this.$init = function () {',
  19073. ' };',
  19074. ' this.$final = function () {',
  19075. ' };',
  19076. ' rtl.addIntf(this, $mod.IUnknown);',
  19077. '});',
  19078. 'this.i = null;',
  19079. 'this.DoGlobal = function (o) {',
  19080. ' rtl.setIntfP($mod, "i", null);',
  19081. ' rtl.setIntfP($mod, "i", rtl.queryIntfT(o, $mod.IUnknown), true);',
  19082. ' rtl.setIntfP($mod, "i", $mod.i);',
  19083. '};',
  19084. 'var k = null;',
  19085. 'this.DoLocal = function (o) {',
  19086. ' var j = null;',
  19087. ' try{',
  19088. ' k = rtl.setIntfL(k, rtl.queryIntfT(o, $mod.IUnknown), true);',
  19089. ' k = rtl.setIntfL(k, $mod.i);',
  19090. ' j = rtl.setIntfL(j, rtl.queryIntfT(o, $mod.IUnknown), true);',
  19091. ' j = rtl.setIntfL(j, $mod.i);',
  19092. ' }finally{',
  19093. ' rtl._Release(j);',
  19094. ' };',
  19095. '};',
  19096. 'this.o = null;',
  19097. '']),
  19098. LinesToStr([ // $mod.$main
  19099. 'rtl.setIntfP($mod, "i", null);',
  19100. 'rtl.setIntfP($mod, "i", rtl.queryIntfT($mod.o, $mod.IUnknown), true);',
  19101. '']));
  19102. end;
  19103. procedure TTestModule.TestClassInterface_COM_AssignArg;
  19104. begin
  19105. StartProgram(false);
  19106. Add([
  19107. '{$interfaces com}',
  19108. 'type',
  19109. ' IUnknown = interface',
  19110. ' function _AddRef: longint;',
  19111. ' function _Release: longint;',
  19112. ' end;',
  19113. ' TObject = class(IUnknown)',
  19114. ' function _AddRef: longint; virtual; abstract;',
  19115. ' function _Release: longint; virtual; abstract;',
  19116. ' end;',
  19117. 'procedure DoDefault(i, j: IUnknown);',
  19118. 'begin',
  19119. ' i:=nil;',
  19120. ' i:=j;',
  19121. 'end;',
  19122. 'begin',
  19123. '']);
  19124. ConvertProgram;
  19125. CheckSource('TestClassInterface_COM_AssignArg',
  19126. LinesToStr([ // statements
  19127. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19128. 'rtl.createClass(this, "TObject", null, function () {',
  19129. ' this.$init = function () {',
  19130. ' };',
  19131. ' this.$final = function () {',
  19132. ' };',
  19133. ' rtl.addIntf(this, $mod.IUnknown);',
  19134. '});',
  19135. 'this.DoDefault = function (i, j) {',
  19136. ' rtl._AddRef(i);',
  19137. ' try {',
  19138. ' i = rtl.setIntfL(i, null);',
  19139. ' i = rtl.setIntfL(i, j);',
  19140. ' } finally {',
  19141. ' rtl._Release(i);',
  19142. ' };',
  19143. '};',
  19144. '']),
  19145. LinesToStr([ // $mod.$main
  19146. '']));
  19147. end;
  19148. procedure TTestModule.TestClassInterface_COM_FunctionResult;
  19149. begin
  19150. StartProgram(false);
  19151. Add([
  19152. '{$interfaces com}',
  19153. 'type',
  19154. ' IUnknown = interface',
  19155. ' function _AddRef: longint;',
  19156. ' function _Release: longint;',
  19157. ' end;',
  19158. ' TObject = class(IUnknown)',
  19159. ' function _AddRef: longint; virtual; abstract;',
  19160. ' function _Release: longint; virtual; abstract;',
  19161. ' end;',
  19162. 'function DoDefault(i: IUnknown): IUnknown;',
  19163. 'begin',
  19164. ' Result:=i;',
  19165. ' if Result<>nil then exit;',
  19166. 'end;',
  19167. 'begin',
  19168. '']);
  19169. ConvertProgram;
  19170. CheckSource('TestClassInterface_COM_FunctionResult',
  19171. LinesToStr([ // statements
  19172. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19173. 'rtl.createClass(this, "TObject", null, function () {',
  19174. ' this.$init = function () {',
  19175. ' };',
  19176. ' this.$final = function () {',
  19177. ' };',
  19178. ' rtl.addIntf(this, $mod.IUnknown);',
  19179. '});',
  19180. 'this.DoDefault = function (i) {',
  19181. ' var Result = null;',
  19182. ' var $ok = false;',
  19183. ' try {',
  19184. ' Result = rtl.setIntfL(Result, i);',
  19185. ' if(Result !== null){',
  19186. ' $ok = true;',
  19187. ' return Result;',
  19188. ' };',
  19189. ' $ok = true;',
  19190. ' } finally {',
  19191. ' if(!$ok) rtl._Release(Result);',
  19192. ' };',
  19193. ' return Result;',
  19194. '};',
  19195. '']),
  19196. LinesToStr([ // $mod.$main
  19197. '']));
  19198. end;
  19199. procedure TTestModule.TestClassInterface_COM_InheritedFuncResult;
  19200. begin
  19201. StartProgram(false);
  19202. Add([
  19203. '{$interfaces com}',
  19204. 'type',
  19205. ' IUnknown = interface',
  19206. ' function _AddRef: longint;',
  19207. ' function _Release: longint;',
  19208. ' end;',
  19209. ' TObject = class(IUnknown)',
  19210. ' function _AddRef: longint; virtual; abstract;',
  19211. ' function _Release: longint; virtual; abstract;',
  19212. ' function GetIntf: IUnknown; virtual;',
  19213. ' end;',
  19214. ' TMouse = class',
  19215. ' function GetIntf: IUnknown; override;',
  19216. ' end;',
  19217. 'function TObject.GetIntf: IUnknown; begin end;',
  19218. 'function TMouse.GetIntf: IUnknown;',
  19219. 'var i: IUnknown;',
  19220. 'begin',
  19221. ' inherited;',
  19222. ' inherited GetIntf;',
  19223. ' inherited GetIntf();',
  19224. ' Result:=inherited GetIntf;',
  19225. ' Result:=inherited GetIntf();',
  19226. ' i:=inherited GetIntf;',
  19227. ' i:=inherited GetIntf();',
  19228. 'end;',
  19229. 'begin',
  19230. '']);
  19231. ConvertProgram;
  19232. CheckSource('TestClassInterface_COM_InheritedFuncResult',
  19233. LinesToStr([ // statements
  19234. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19235. 'rtl.createClass(this, "TObject", null, function () {',
  19236. ' this.$init = function () {',
  19237. ' };',
  19238. ' this.$final = function () {',
  19239. ' };',
  19240. ' this.GetIntf = function () {',
  19241. ' var Result = null;',
  19242. ' return Result;',
  19243. ' };',
  19244. ' rtl.addIntf(this, $mod.IUnknown);',
  19245. '});',
  19246. 'rtl.createClass(this, "TMouse", this.TObject, function () {',
  19247. ' this.GetIntf = function () {',
  19248. ' var Result = null;',
  19249. ' var i = null;',
  19250. ' var $ir = rtl.createIntfRefs();',
  19251. ' var $ok = false;',
  19252. ' try {',
  19253. ' $ir.ref(1, $mod.TObject.GetIntf.call(this));',
  19254. ' $ir.ref(2, $mod.TObject.GetIntf.call(this));',
  19255. ' $ir.ref(3, $mod.TObject.GetIntf.call(this));',
  19256. ' Result = rtl.setIntfL(Result, $mod.TObject.GetIntf.call(this), true);',
  19257. ' Result = rtl.setIntfL(Result, $mod.TObject.GetIntf.call(this), true);',
  19258. ' i = rtl.setIntfL(i, $mod.TObject.GetIntf.call(this), true);',
  19259. ' i = rtl.setIntfL(i, $mod.TObject.GetIntf.call(this), true);',
  19260. ' $ok = true;',
  19261. ' } finally {',
  19262. ' $ir.free();',
  19263. ' rtl._Release(i);',
  19264. ' if (!$ok) rtl._Release(Result);',
  19265. ' };',
  19266. ' return Result;',
  19267. ' };',
  19268. ' rtl.addIntf(this, $mod.IUnknown);',
  19269. '});',
  19270. '']),
  19271. LinesToStr([ // $mod.$main
  19272. '']));
  19273. end;
  19274. procedure TTestModule.TestClassInterface_COM_IsAsTypeCasts;
  19275. begin
  19276. StartProgram(false);
  19277. Add([
  19278. '{$interfaces com}',
  19279. 'type',
  19280. ' IUnknown = interface',
  19281. ' function _AddRef: longint;',
  19282. ' function _Release: longint;',
  19283. ' end;',
  19284. ' TObject = class(IUnknown)',
  19285. ' function _AddRef: longint; virtual; abstract;',
  19286. ' function _Release: longint; virtual; abstract;',
  19287. ' end;',
  19288. 'procedure DoDefault(i, j: IUnknown; o: TObject);',
  19289. 'begin',
  19290. ' if i is IUnknown then ;',
  19291. ' if o is IUnknown then ;',
  19292. ' if i is TObject then ;',
  19293. ' i:=j as IUnknown;',
  19294. ' i:=o as IUnknown;',
  19295. ' o:=j as TObject;',
  19296. ' i:=IUnknown(j);',
  19297. ' i:=IUnknown(o);',
  19298. ' o:=TObject(i);',
  19299. 'end;',
  19300. 'begin',
  19301. '']);
  19302. ConvertProgram;
  19303. CheckSource('TestClassInterface_COM_IsAsTypeCasts',
  19304. LinesToStr([ // statements
  19305. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19306. 'rtl.createClass(this, "TObject", null, function () {',
  19307. ' this.$init = function () {',
  19308. ' };',
  19309. ' this.$final = function () {',
  19310. ' };',
  19311. ' rtl.addIntf(this, $mod.IUnknown);',
  19312. '});',
  19313. 'this.DoDefault = function (i, j, o) {',
  19314. ' rtl._AddRef(i);',
  19315. ' try {',
  19316. ' if (rtl.intfIsIntfT(i, $mod.IUnknown)) ;',
  19317. ' if (rtl.queryIntfIsT(o, $mod.IUnknown)) ;',
  19318. ' if (rtl.intfIsClass(i, $mod.TObject)) ;',
  19319. ' i = rtl.setIntfL(i, rtl.intfAsIntfT(j, $mod.IUnknown));',
  19320. ' i = rtl.setIntfL(i, rtl.queryIntfT(o, $mod.IUnknown), true);',
  19321. ' o = rtl.intfAsClass(j, $mod.TObject);',
  19322. ' i = rtl.setIntfL(i, j);',
  19323. ' i = rtl.setIntfL(i, rtl.queryIntfT(o, $mod.IUnknown), true);',
  19324. ' o = rtl.intfToClass(i, $mod.TObject);',
  19325. ' } finally {',
  19326. ' rtl._Release(i);',
  19327. ' };',
  19328. '};',
  19329. '']),
  19330. LinesToStr([ // $mod.$main
  19331. '']));
  19332. end;
  19333. procedure TTestModule.TestClassInterface_COM_PassAsArg;
  19334. begin
  19335. StartProgram(false);
  19336. Add([
  19337. '{$interfaces com}',
  19338. 'type',
  19339. ' IUnknown = interface',
  19340. ' function _AddRef: longint;',
  19341. ' function _Release: longint;',
  19342. ' end;',
  19343. ' TObject = class(IUnknown)',
  19344. ' function _AddRef: longint; virtual; abstract;',
  19345. ' function _Release: longint; virtual; abstract;',
  19346. ' end;',
  19347. 'procedure DoIt(v: IUnknown; const j: IUnknown; var k: IUnknown; out l: IUnknown);',
  19348. 'var o: TObject;',
  19349. 'begin',
  19350. ' DoIt(v,v,v,v);',
  19351. ' DoIt(o,o,k,k);',
  19352. 'end;',
  19353. 'procedure DoSome;',
  19354. 'var v: IUnknown;',
  19355. 'begin',
  19356. ' DoIt(v,v,v,v);',
  19357. 'end;',
  19358. 'var i: IUnknown;',
  19359. 'begin',
  19360. ' DoIt(i,i,i,i);',
  19361. '']);
  19362. ConvertProgram;
  19363. CheckSource('TestClassInterface_COM_PassAsArg',
  19364. LinesToStr([ // statements
  19365. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19366. 'rtl.createClass(this, "TObject", null, function () {',
  19367. ' this.$init = function () {',
  19368. ' };',
  19369. ' this.$final = function () {',
  19370. ' };',
  19371. ' rtl.addIntf(this, $mod.IUnknown);',
  19372. '});',
  19373. 'this.DoIt = function (v, j, k, l) {',
  19374. ' var o = null;',
  19375. ' var $ir = rtl.createIntfRefs();',
  19376. ' rtl._AddRef(v);',
  19377. ' try {',
  19378. ' $mod.DoIt(v, v, {',
  19379. ' get: function () {',
  19380. ' return v;',
  19381. ' },',
  19382. ' set: function (w) {',
  19383. ' v = rtl.setIntfL(v, w);',
  19384. ' }',
  19385. ' }, {',
  19386. ' get: function () {',
  19387. ' return v;',
  19388. ' },',
  19389. ' set: function (w) {',
  19390. ' v = rtl.setIntfL(v, w);',
  19391. ' }',
  19392. ' });',
  19393. ' $mod.DoIt($ir.ref(1, rtl.queryIntfT(o, $mod.IUnknown)), $ir.ref(2, rtl.queryIntfT(o, $mod.IUnknown)), k, k);',
  19394. ' } finally {',
  19395. ' $ir.free();',
  19396. ' rtl._Release(v);',
  19397. ' };',
  19398. '};',
  19399. 'this.DoSome = function () {',
  19400. ' var v = null;',
  19401. ' try {',
  19402. ' $mod.DoIt(v, v, {',
  19403. ' get: function () {',
  19404. ' return v;',
  19405. ' },',
  19406. ' set: function (w) {',
  19407. ' v = rtl.setIntfL(v, w);',
  19408. ' }',
  19409. ' }, {',
  19410. ' get: function () {',
  19411. ' return v;',
  19412. ' },',
  19413. ' set: function (w) {',
  19414. ' v = rtl.setIntfL(v, w);',
  19415. ' }',
  19416. ' });',
  19417. ' } finally {',
  19418. ' rtl._Release(v);',
  19419. ' };',
  19420. '};',
  19421. 'this.i = null;',
  19422. '']),
  19423. LinesToStr([ // $mod.$main
  19424. '$mod.DoIt($mod.i, $mod.i, {',
  19425. ' p: $mod,',
  19426. ' get: function () {',
  19427. ' return this.p.i;',
  19428. ' },',
  19429. ' set: function (v) {',
  19430. ' rtl.setIntfP(this.p, "i", v);',
  19431. ' }',
  19432. '}, {',
  19433. ' p: $mod,',
  19434. ' get: function () {',
  19435. ' return this.p.i;',
  19436. ' },',
  19437. ' set: function (v) {',
  19438. ' rtl.setIntfP(this.p, "i", v);',
  19439. ' }',
  19440. '});',
  19441. '']));
  19442. end;
  19443. procedure TTestModule.TestClassInterface_COM_PassToUntypedParam;
  19444. begin
  19445. StartProgram(false);
  19446. Add([
  19447. '{$interfaces com}',
  19448. 'type',
  19449. ' IUnknown = interface',
  19450. ' function _AddRef: longint;',
  19451. ' function _Release: longint;',
  19452. ' end;',
  19453. ' TObject = class(IUnknown)',
  19454. ' function _AddRef: longint; virtual; abstract;',
  19455. ' function _Release: longint; virtual; abstract;',
  19456. ' end;',
  19457. 'procedure DoIt(out i);',
  19458. 'begin end;',
  19459. 'procedure DoSome;',
  19460. 'var v: IUnknown;',
  19461. 'begin',
  19462. ' DoIt(v);',
  19463. 'end;',
  19464. 'function GetIt: IUnknown;',
  19465. 'begin',
  19466. ' DoIt(Result);',
  19467. 'end;',
  19468. 'var i: IUnknown;',
  19469. 'begin',
  19470. ' DoIt(i);',
  19471. '']);
  19472. ConvertProgram;
  19473. CheckSource('TestClassInterface_COM_PassToUntypedParam',
  19474. LinesToStr([ // statements
  19475. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19476. 'rtl.createClass(this, "TObject", null, function () {',
  19477. ' this.$init = function () {',
  19478. ' };',
  19479. ' this.$final = function () {',
  19480. ' };',
  19481. ' rtl.addIntf(this, $mod.IUnknown);',
  19482. '});',
  19483. 'this.DoIt = function (i) {',
  19484. '};',
  19485. 'this.DoSome = function () {',
  19486. ' var v = null;',
  19487. ' try {',
  19488. ' $mod.DoIt({',
  19489. ' get: function () {',
  19490. ' return v;',
  19491. ' },',
  19492. ' set: function (w) {',
  19493. ' v = w;',
  19494. ' }',
  19495. ' });',
  19496. ' } finally {',
  19497. ' rtl._Release(v);',
  19498. ' };',
  19499. '};',
  19500. 'this.GetIt = function () {',
  19501. ' var Result = null;',
  19502. ' var $ok = false;',
  19503. ' try {',
  19504. ' $mod.DoIt({',
  19505. ' get: function () {',
  19506. ' return Result;',
  19507. ' },',
  19508. ' set: function (v) {',
  19509. ' Result = v;',
  19510. ' }',
  19511. ' });',
  19512. ' $ok = true;',
  19513. ' } finally {',
  19514. ' if (!$ok) rtl._Release(Result);',
  19515. ' };',
  19516. ' return Result;',
  19517. '};',
  19518. 'this.i = null;',
  19519. '']),
  19520. LinesToStr([ // $mod.$main
  19521. 'try {',
  19522. ' $mod.DoIt({',
  19523. ' p: $mod,',
  19524. ' get: function () {',
  19525. ' return this.p.i;',
  19526. ' },',
  19527. ' set: function (v) {',
  19528. ' this.p.i = v;',
  19529. ' }',
  19530. ' });',
  19531. '} finally {',
  19532. ' rtl._Release($mod.i);',
  19533. '};',
  19534. '']));
  19535. end;
  19536. procedure TTestModule.TestClassInterface_COM_FunctionInExpr;
  19537. begin
  19538. StartProgram(false);
  19539. Add([
  19540. '{$interfaces com}',
  19541. 'type',
  19542. ' IUnknown = interface',
  19543. ' function _AddRef: longint;',
  19544. ' function _Release: longint;',
  19545. ' end;',
  19546. ' TObject = class(IUnknown)',
  19547. ' function _AddRef: longint; virtual; abstract;',
  19548. ' function _Release: longint; virtual; abstract;',
  19549. ' end;',
  19550. 'function GetIt: IUnknown;',
  19551. 'begin',
  19552. 'end;',
  19553. 'procedure DoSome;',
  19554. 'var v: IUnknown;',
  19555. ' i: longint;',
  19556. 'begin',
  19557. ' v:=GetIt;',
  19558. ' v:=GetIt();',
  19559. ' GetIt()._AddRef;',
  19560. ' i:=GetIt()._AddRef;',
  19561. 'end;',
  19562. 'var v: IUnknown;',
  19563. ' i: longint;',
  19564. 'begin',
  19565. ' v:=GetIt;',
  19566. ' v:=GetIt();',
  19567. ' GetIt()._AddRef;',
  19568. ' i:=GetIt()._AddRef;',
  19569. '']);
  19570. ConvertProgram;
  19571. CheckSource('TestClassInterface_COM_FunctionInExpr',
  19572. LinesToStr([ // statements
  19573. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19574. 'rtl.createClass(this, "TObject", null, function () {',
  19575. ' this.$init = function () {',
  19576. ' };',
  19577. ' this.$final = function () {',
  19578. ' };',
  19579. ' rtl.addIntf(this, $mod.IUnknown);',
  19580. '});',
  19581. 'this.GetIt = function () {',
  19582. ' var Result = null;',
  19583. ' return Result;',
  19584. '};',
  19585. 'this.DoSome = function () {',
  19586. ' var v = null;',
  19587. ' var i = 0;',
  19588. ' var $ir = rtl.createIntfRefs();',
  19589. ' try {',
  19590. ' v = rtl.setIntfL(v, $mod.GetIt(), true);',
  19591. ' v = rtl.setIntfL(v, $mod.GetIt(), true);',
  19592. ' $ir.ref(1, $mod.GetIt())._AddRef();',
  19593. ' i = $ir.ref(2, $mod.GetIt())._AddRef();',
  19594. ' } finally {',
  19595. ' $ir.free();',
  19596. ' rtl._Release(v);',
  19597. ' };',
  19598. '};',
  19599. 'this.v = null;',
  19600. 'this.i = 0;',
  19601. '']),
  19602. LinesToStr([ // $mod.$main
  19603. 'var $ir = rtl.createIntfRefs();',
  19604. 'try {',
  19605. ' rtl.setIntfP($mod, "v", $mod.GetIt(), true);',
  19606. ' rtl.setIntfP($mod, "v", $mod.GetIt(), true);',
  19607. ' $ir.ref(1, $mod.GetIt())._AddRef();',
  19608. ' $mod.i = $ir.ref(2, $mod.GetIt())._AddRef();',
  19609. '} finally {',
  19610. ' $ir.free();',
  19611. '};',
  19612. '']));
  19613. end;
  19614. procedure TTestModule.TestClassInterface_COM_Property;
  19615. begin
  19616. StartProgram(false);
  19617. Add([
  19618. '{$interfaces com}',
  19619. 'type',
  19620. ' IUnknown = interface',
  19621. ' function _AddRef: longint;',
  19622. ' function _Release: longint;',
  19623. ' end;',
  19624. ' TObject = class(IUnknown)',
  19625. ' FAnt: IUnknown;',
  19626. ' function _AddRef: longint; virtual; abstract;',
  19627. ' function _Release: longint; virtual; abstract;',
  19628. ' function GetBird: IUnknown; virtual; abstract;',
  19629. ' procedure SetBird(Value: IUnknown); virtual; abstract;',
  19630. ' function GetItems(Index: longint): IUnknown; virtual; abstract;',
  19631. ' procedure SetItems(Index: longint; Value: IUnknown); virtual; abstract;',
  19632. ' property Ant: IUnknown read FAnt write FAnt;',
  19633. ' property Bird: IUnknown read GetBird write SetBird;',
  19634. ' property Items[Index: longint]: IUnknown read GetItems write SetItems; default;',
  19635. ' end;',
  19636. 'procedure DoIt;',
  19637. 'var',
  19638. ' o: TObject;',
  19639. ' v: IUnknown;',
  19640. 'begin',
  19641. ' v:=o.Ant;',
  19642. ' o.Ant:=v;',
  19643. ' o.Ant:=o.Ant;',
  19644. ' v:=o.Bird;',
  19645. ' o.Bird:=v;',
  19646. ' o.Bird:=o.Bird;',
  19647. ' v:=o.Items[1];',
  19648. ' o.Items[2]:=v;',
  19649. ' o.Items[3]:=o.Items[4];',
  19650. ' v:=o[5];',
  19651. ' o[6]:=v;',
  19652. ' o[7]:=o[8];',
  19653. 'end;',
  19654. 'begin',
  19655. '']);
  19656. ConvertProgram;
  19657. CheckSource('TestClassInterface_COM_Property',
  19658. LinesToStr([ // statements
  19659. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19660. 'rtl.createClass(this, "TObject", null, function () {',
  19661. ' this.$init = function () {',
  19662. ' this.FAnt = null;',
  19663. ' };',
  19664. ' this.$final = function () {',
  19665. ' this.FAnt = undefined;',
  19666. ' };',
  19667. ' rtl.addIntf(this, $mod.IUnknown);',
  19668. '});',
  19669. 'this.DoIt = function () {',
  19670. ' var o = null;',
  19671. ' var v = null;',
  19672. ' var $ir = rtl.createIntfRefs();',
  19673. ' try {',
  19674. ' v = rtl.setIntfL(v, o.FAnt);',
  19675. ' rtl.setIntfP(o, "FAnt", v);',
  19676. ' rtl.setIntfP(o, "FAnt", o.FAnt);',
  19677. ' v = rtl.setIntfL(v, o.GetBird(), true);',
  19678. ' o.SetBird(v);',
  19679. ' o.SetBird($ir.ref(1, o.GetBird()));',
  19680. ' v = rtl.setIntfL(v, o.GetItems(1), true);',
  19681. ' o.SetItems(2, v);',
  19682. ' o.SetItems(3, $ir.ref(2, o.GetItems(4)));',
  19683. ' v = rtl.setIntfL(v, o.GetItems(5), true);',
  19684. ' o.SetItems(6, v);',
  19685. ' o.SetItems(7, $ir.ref(3, o.GetItems(8)));',
  19686. ' } finally {',
  19687. ' $ir.free();',
  19688. ' rtl._Release(v);',
  19689. ' };',
  19690. '};',
  19691. '']),
  19692. LinesToStr([ // $mod.$main
  19693. '']));
  19694. end;
  19695. procedure TTestModule.TestClassInterface_COM_IntfProperty;
  19696. begin
  19697. StartProgram(false);
  19698. Add([
  19699. '{$interfaces com}',
  19700. 'type',
  19701. ' IUnknown = interface',
  19702. ' function _AddRef: longint;',
  19703. ' function _Release: longint;',
  19704. ' function GetBird: IUnknown;',
  19705. ' procedure SetBird(Value: IUnknown);',
  19706. ' function GetItems(Index: longint): IUnknown;',
  19707. ' procedure SetItems(Index: longint; Value: IUnknown);',
  19708. ' property Bird: IUnknown read GetBird write SetBird;',
  19709. ' property Items[Index: longint]: IUnknown read GetItems write SetItems; default;',
  19710. ' end;',
  19711. ' TObject = class(IUnknown)',
  19712. ' function _AddRef: longint; virtual; abstract;',
  19713. ' function _Release: longint; virtual; abstract;',
  19714. ' function GetBird: IUnknown; virtual; abstract;',
  19715. ' procedure SetBird(Value: IUnknown); virtual; abstract;',
  19716. ' function GetItems(Index: longint): IUnknown; virtual; abstract;',
  19717. ' procedure SetItems(Index: longint; Value: IUnknown); virtual; abstract;',
  19718. ' end;',
  19719. 'procedure DoIt;',
  19720. 'var',
  19721. ' o: TObject;',
  19722. ' v: IUnknown;',
  19723. 'begin',
  19724. ' v:=v.Items[1];',
  19725. ' v.Items[2]:=v;',
  19726. ' v.Items[3]:=v.Items[4];',
  19727. ' v:=v[5];',
  19728. ' v[6]:=v;',
  19729. ' v[7]:=v[8];',
  19730. ' v[9].Bird.Bird:=v;',
  19731. ' v:=v.Bird[10].Bird',
  19732. 'end;',
  19733. 'begin',
  19734. '']);
  19735. ConvertProgram;
  19736. CheckSource('TestClassInterface_COM_IntfProperty',
  19737. LinesToStr([ // statements
  19738. 'rtl.createInterface(this, "IUnknown", "{385F5482-571B-338C-8130-4E97F330543B}", [',
  19739. ' "_AddRef",',
  19740. ' "_Release",',
  19741. ' "GetBird",',
  19742. ' "SetBird",',
  19743. ' "GetItems",',
  19744. ' "SetItems"',
  19745. '], null);',
  19746. 'rtl.createClass(this, "TObject", null, function () {',
  19747. ' this.$init = function () {',
  19748. ' };',
  19749. ' this.$final = function () {',
  19750. ' };',
  19751. ' rtl.addIntf(this, $mod.IUnknown);',
  19752. '});',
  19753. 'this.DoIt = function () {',
  19754. ' var o = null;',
  19755. ' var v = null;',
  19756. ' var $ir = rtl.createIntfRefs();',
  19757. ' try {',
  19758. ' v = rtl.setIntfL(v, v.GetItems(1), true);',
  19759. ' v.SetItems(2, v);',
  19760. ' v.SetItems(3, $ir.ref(1, v.GetItems(4)));',
  19761. ' v = rtl.setIntfL(v, v.GetItems(5), true);',
  19762. ' v.SetItems(6, v);',
  19763. ' v.SetItems(7, $ir.ref(2, v.GetItems(8)));',
  19764. ' $ir.ref(4, $ir.ref(3, v.GetItems(9)).GetBird()).SetBird(v);',
  19765. ' v = rtl.setIntfL(v, $ir.ref(6, $ir.ref(5, v.GetBird()).GetItems(10)).GetBird(), true);',
  19766. ' } finally {',
  19767. ' $ir.free();',
  19768. ' rtl._Release(v);',
  19769. ' };',
  19770. '};',
  19771. '']),
  19772. LinesToStr([ // $mod.$main
  19773. '']));
  19774. end;
  19775. procedure TTestModule.TestClassInterface_COM_Delegation;
  19776. begin
  19777. StartProgram(false);
  19778. Add([
  19779. '{$interfaces com}',
  19780. 'type',
  19781. ' IUnknown = interface',
  19782. ' function _AddRef: longint;',
  19783. ' function _Release: longint;',
  19784. ' end;',
  19785. ' IBird = interface(IUnknown)',
  19786. ' procedure Fly(s: string);',
  19787. ' end;',
  19788. ' IEagle = interface(IBird) end;',
  19789. ' IDove = interface(IBird) end;',
  19790. ' ISwallow = interface(IBird) end;',
  19791. ' TObject = class',
  19792. ' end;',
  19793. ' TBird = class(TObject,IBird,IEagle,IDove,ISwallow)',
  19794. ' function _AddRef: longint; virtual; abstract;',
  19795. ' function _Release: longint; virtual; abstract;',
  19796. ' procedure Fly(s: string); virtual; abstract;',
  19797. ' end;',
  19798. ' TBat = class(IBird,IEagle,IDove,ISwallow)',
  19799. ' function _AddRef: longint; virtual; abstract;',
  19800. ' function _Release: longint; virtual; abstract;',
  19801. ' FBirdIntf: IBird;',
  19802. ' property BirdIntf: IBird read FBirdIntf implements IBird;',
  19803. ' function GetEagleIntf: IEagle; virtual; abstract;',
  19804. ' property EagleIntf: IEagle read GetEagleIntf implements IEagle;',
  19805. ' FDoveObj: TBird;',
  19806. ' property DoveObj: TBird read FDoveObj implements IDove;',
  19807. ' function GetSwallowObj: TBird; virtual; abstract;',
  19808. ' property SwallowObj: TBird read GetSwallowObj implements ISwallow;',
  19809. ' end;',
  19810. 'begin',
  19811. '']);
  19812. ConvertProgram;
  19813. CheckSource('TestClassInterface_COM_Delegation',
  19814. LinesToStr([ // statements
  19815. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  19816. 'rtl.createInterface(this, "IBird", "{CC440C7F-7623-3DEE-AE88-000B86AAF108}", ["Fly"], this.IUnknown);',
  19817. 'rtl.createInterface(this, "IEagle", "{4B6A41C9-B020-3D7C-B688-96D19022B1B4}", [], this.IBird);',
  19818. 'rtl.createInterface(this, "IDove", "{4B6A41C9-B020-3D7C-B688-96D18EF16074}", [], this.IBird);',
  19819. 'rtl.createInterface(this, "ISwallow", "{BB6A41C9-B020-3D7C-B688-96D1CBDCB359}", [], this.IBird);',
  19820. 'rtl.createClass(this, "TObject", null, function () {',
  19821. ' this.$init = function () {',
  19822. ' };',
  19823. ' this.$final = function () {',
  19824. ' };',
  19825. '});',
  19826. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  19827. ' rtl.addIntf(this, $mod.IBird);',
  19828. ' rtl.addIntf(this, $mod.IEagle);',
  19829. ' rtl.addIntf(this, $mod.IDove);',
  19830. ' rtl.addIntf(this, $mod.ISwallow);',
  19831. '});',
  19832. 'rtl.createClass(this, "TBat", this.TObject, function () {',
  19833. ' this.$init = function () {',
  19834. ' $mod.TObject.$init.call(this);',
  19835. ' this.FBirdIntf = null;',
  19836. ' this.FDoveObj = null;',
  19837. ' };',
  19838. ' this.$final = function () {',
  19839. ' this.FBirdIntf = undefined;',
  19840. ' this.FDoveObj = undefined;',
  19841. ' $mod.TObject.$final.call(this);',
  19842. ' };',
  19843. ' this.$intfmaps = {',
  19844. ' "{CC440C7F-7623-3DEE-AE88-000B86AAF108}": function () {',
  19845. ' return rtl._AddRef(this.FBirdIntf);',
  19846. ' },',
  19847. ' "{4B6A41C9-B020-3D7C-B688-96D19022B1B4}": function () {',
  19848. ' return this.GetEagleIntf();',
  19849. ' },',
  19850. ' "{4B6A41C9-B020-3D7C-B688-96D18EF16074}": function () {',
  19851. ' return rtl.queryIntfT(this.FDoveObj, $mod.IDove);',
  19852. ' },',
  19853. ' "{BB6A41C9-B020-3D7C-B688-96D1CBDCB359}": function () {',
  19854. ' return rtl.queryIntfT(this.GetSwallowObj(), $mod.ISwallow);',
  19855. ' }',
  19856. ' };',
  19857. '});',
  19858. '']),
  19859. LinesToStr([ // $mod.$main
  19860. '']));
  19861. end;
  19862. procedure TTestModule.TestClassInterface_COM_With;
  19863. begin
  19864. StartProgram(false);
  19865. Add([
  19866. '{$interfaces com}',
  19867. 'type',
  19868. ' IUnknown = interface',
  19869. ' function _AddRef: longint;',
  19870. ' function _Release: longint;',
  19871. ' function GetAnt: IUnknown;',
  19872. ' property Ant: IUnknown read GetAnt;',
  19873. ' end;',
  19874. ' TObject = class(IUnknown)',
  19875. ' function _AddRef: longint; virtual; abstract;',
  19876. ' function _Release: longint; virtual; abstract;',
  19877. ' function GetAnt: IUnknown; virtual; abstract;',
  19878. ' property Ant: IUnknown read GetAnt;',
  19879. ' end;',
  19880. 'procedure DoIt;',
  19881. 'var',
  19882. ' i: IUnknown;',
  19883. 'begin',
  19884. ' with i do ',
  19885. ' GetAnt;',
  19886. ' with i.Ant, Ant do ',
  19887. ' GetAnt;',
  19888. 'end;',
  19889. 'begin',
  19890. '']);
  19891. ConvertProgram;
  19892. CheckSource('TestClassInterface_COM_With',
  19893. LinesToStr([ // statements
  19894. 'rtl.createInterface(this, "IUnknown", "{D7ADB00D-C6B6-39FB-BDDF-21CD521DDFA9}", ["_AddRef", "_Release", "GetAnt"], null);',
  19895. 'rtl.createClass(this, "TObject", null, function () {',
  19896. ' this.$init = function () {',
  19897. ' };',
  19898. ' this.$final = function () {',
  19899. ' };',
  19900. ' rtl.addIntf(this, $mod.IUnknown);',
  19901. '});',
  19902. 'this.DoIt = function () {',
  19903. ' var i = null;',
  19904. ' var $ir = rtl.createIntfRefs();',
  19905. ' try {',
  19906. ' $ir.ref(1, i.GetAnt());',
  19907. ' var $with = $ir.ref(2, i.GetAnt());',
  19908. ' var $with1 = $ir.ref(3, $with.GetAnt());',
  19909. ' $ir.ref(4, $with1.GetAnt());',
  19910. ' } finally {',
  19911. ' $ir.free();',
  19912. ' };',
  19913. '};',
  19914. '']),
  19915. LinesToStr([ // $mod.$main
  19916. '']));
  19917. end;
  19918. procedure TTestModule.TestClassInterface_COM_ForIn;
  19919. begin
  19920. StartProgram(false);
  19921. Add([
  19922. '{$interfaces com}',
  19923. 'type',
  19924. ' IUnknown = interface end;',
  19925. ' TObject = class',
  19926. ' Id: longint;',
  19927. ' end;',
  19928. ' IEnumerator = interface(IUnknown)',
  19929. ' function GetCurrent: TObject;',
  19930. ' function MoveNext: Boolean;',
  19931. ' property Current: TObject read GetCurrent;',
  19932. ' end;',
  19933. ' IEnumerable = interface(IUnknown)',
  19934. ' function GetEnumerator: IEnumerator;',
  19935. ' end;',
  19936. 'var',
  19937. ' o: TObject;',
  19938. ' i: IEnumerable;',
  19939. 'begin',
  19940. ' for o in i do o.Id:=3;',
  19941. '']);
  19942. ConvertProgram;
  19943. CheckSource('TestClassInterface_COM_ForIn',
  19944. LinesToStr([ // statements
  19945. 'rtl.createInterface(this, "IUnknown", "{B92D5841-758A-322B-B800-000000000000}", [], null);',
  19946. 'rtl.createClass(this, "TObject", null, function () {',
  19947. ' this.$init = function () {',
  19948. ' this.Id = 0;',
  19949. ' };',
  19950. ' this.$final = function () {',
  19951. ' };',
  19952. '});',
  19953. 'rtl.createInterface(this, "IEnumerator", "{95D7745D-ED61-3F13-BBE4-07708161999E}", ["GetCurrent", "MoveNext"], this.IUnknown);',
  19954. 'rtl.createInterface(this, "IEnumerable", "{8CC9D45D-ED7D-3B73-96B6-290B931BB19E}", ["GetEnumerator"], this.IUnknown);',
  19955. 'this.o = null;',
  19956. 'this.i = null;',
  19957. '']),
  19958. LinesToStr([ // $mod.$main
  19959. 'var $in = $mod.i.GetEnumerator();',
  19960. 'try {',
  19961. ' while ($in.MoveNext()) {',
  19962. ' $mod.o = $in.GetCurrent();',
  19963. ' $mod.o.Id = 3;',
  19964. ' }',
  19965. '} finally {',
  19966. ' rtl._Release($in)',
  19967. '};',
  19968. '']));
  19969. end;
  19970. procedure TTestModule.TestClassInterface_COM_ArrayOfIntfFail;
  19971. begin
  19972. StartProgram(false);
  19973. Add([
  19974. '{$interfaces com}',
  19975. 'type',
  19976. ' IUnknown = interface',
  19977. ' function _AddRef: longint;',
  19978. ' function _Release: longint;',
  19979. ' end;',
  19980. ' TObject = class',
  19981. ' end;',
  19982. ' TArrOfIntf = array of IUnknown;',
  19983. 'begin',
  19984. '']);
  19985. SetExpectedPasResolverError('Not supported: array of COM-interface',nNotSupportedX);
  19986. ConvertProgram;
  19987. end;
  19988. procedure TTestModule.TestClassInterface_COM_RecordIntfFail;
  19989. begin
  19990. StartProgram(false);
  19991. Add([
  19992. '{$interfaces com}',
  19993. 'type',
  19994. ' IUnknown = interface',
  19995. ' function _AddRef: longint;',
  19996. ' function _Release: longint;',
  19997. ' end;',
  19998. ' TRec = record',
  19999. ' i: IUnknown;',
  20000. ' end;',
  20001. 'begin',
  20002. '']);
  20003. SetExpectedPasResolverError('Not supported: COM-interface as record member',nNotSupportedX);
  20004. ConvertProgram;
  20005. end;
  20006. procedure TTestModule.TestClassInterface_COM_UnitInitialization;
  20007. begin
  20008. StartUnit(false);
  20009. Add([
  20010. '{$interfaces com}',
  20011. 'interface',
  20012. 'implementation',
  20013. 'type',
  20014. ' IUnknown = interface',
  20015. ' function _AddRef: longint;',
  20016. ' end;',
  20017. ' TObject = class(IUnknown)',
  20018. ' function _AddRef: longint;',
  20019. ' end;',
  20020. 'function TObject._AddRef: longint; begin end;',
  20021. 'var i: IUnknown;',
  20022. ' o: TObject;',
  20023. 'initialization',
  20024. ' i:=nil;',
  20025. ' i:=i;',
  20026. ' i:=o;',
  20027. ' if (o as IUnknown)=nil then ;',
  20028. '']);
  20029. ConvertUnit;
  20030. CheckSource('TestClassInterface_COM_UnitInitialization',
  20031. LinesToStr([ // statements
  20032. 'var $impl = $mod.$impl;',
  20033. '']),
  20034. LinesToStr([ // this.$init
  20035. 'var $ir = rtl.createIntfRefs();',
  20036. 'try {',
  20037. ' rtl.setIntfP($impl, "i", null);',
  20038. ' rtl.setIntfP($impl, "i", $impl.i);',
  20039. ' rtl.setIntfP($impl, "i", rtl.queryIntfT($impl.o, $impl.IUnknown), true);',
  20040. ' if ($ir.ref(1, rtl.queryIntfT($impl.o, $impl.IUnknown)) === null) ;',
  20041. '} finally {',
  20042. ' $ir.free();',
  20043. '};',
  20044. '']),
  20045. LinesToStr([ // implementation
  20046. 'rtl.createInterface($impl, "IUnknown", "{B92D5841-758A-322B-BDDF-21CD52180000}", ["_AddRef"], null);',
  20047. 'rtl.createClass($impl, "TObject", null, function () {',
  20048. ' this.$init = function () {',
  20049. ' };',
  20050. ' this.$final = function () {',
  20051. ' };',
  20052. ' this._AddRef = function () {',
  20053. ' var Result = 0;',
  20054. ' return Result;',
  20055. ' };',
  20056. ' rtl.addIntf(this, $impl.IUnknown);',
  20057. '});',
  20058. '$impl.i = null;',
  20059. '$impl.o = null;',
  20060. ''])
  20061. );
  20062. end;
  20063. procedure TTestModule.TestClassInterface_GUID;
  20064. begin
  20065. StartProgram(false);
  20066. Add([
  20067. '{$interfaces corba}',
  20068. 'type',
  20069. ' IUnknown = interface',
  20070. ' [''{f31db68f-3010-D355-4EBA-CDD4EF4A737C}'']',
  20071. ' end;',
  20072. ' TObject = class end;',
  20073. ' TGUID = record D1, D2, D3, D4: word; end;',
  20074. ' TAliasGUID = TGUID;',
  20075. ' TGUIDString = type string;',
  20076. ' TAliasGUIDString = TGUIDString;',
  20077. 'procedure DoConstGUIDIt(const g: TAliasGUID); overload;',
  20078. 'begin end;',
  20079. 'procedure DoDefGUID(g: TAliasGUID); overload;',
  20080. 'begin end;',
  20081. 'procedure DoStr(const s: TAliasGUIDString); overload;',
  20082. 'begin end;',
  20083. 'var',
  20084. ' i: IUnknown;',
  20085. ' g: TAliasGUID = ''{d91c9af4-3C93-420F-A303-BF5BA82BFD23}'';',
  20086. ' s: TAliasGUIDString;',
  20087. 'begin',
  20088. ' DoConstGUIDIt(IUnknown);',
  20089. ' DoDefGUID(IUnknown);',
  20090. ' DoStr(IUnknown);',
  20091. ' DoConstGUIDIt(i);',
  20092. ' DoDefGUID(i);',
  20093. ' DoStr(i);',
  20094. ' DoConstGUIDIt(''{D91C9AF4-3c93-420f-A303-BF5BA82BFD23}'');',
  20095. ' DoDefGUID(''{D91C9AF4-3c93-420f-A303-BF5BA82BFD23}'');',
  20096. ' DoStr(g);',
  20097. ' g:=i;',
  20098. ' g:=IUnknown;',
  20099. ' g:=''{D91C9AF4-3C93-420F-A303-bf5ba82bfd23}'';',
  20100. ' s:=i;',
  20101. ' s:=IUnknown;',
  20102. ' s:=g;',
  20103. ' if g=i then ;',
  20104. ' if i=g then ;',
  20105. ' if g=IUnknown then ;',
  20106. ' if IUnknown=g then ;',
  20107. ' if s=i then ;',
  20108. ' if i=s then ;',
  20109. ' if s=IUnknown then ;',
  20110. ' if IUnknown=s then ;',
  20111. ' if s=g then ;',
  20112. ' if g=s then ;',
  20113. '']);
  20114. ConvertProgram;
  20115. CheckSource('TestClassInterface_GUID',
  20116. LinesToStr([ // statements
  20117. 'rtl.createInterface(this, "IUnknown", "{F31DB68F-3010-D355-4EBA-CDD4EF4A737C}", [], null);',
  20118. 'rtl.createClass(this, "TObject", null, function () {',
  20119. ' this.$init = function () {',
  20120. ' };',
  20121. ' this.$final = function () {',
  20122. ' };',
  20123. '});',
  20124. 'rtl.recNewT(this, "TGUID", function () {',
  20125. ' this.D1 = 0;',
  20126. ' this.D2 = 0;',
  20127. ' this.D3 = 0;',
  20128. ' this.D4 = 0;',
  20129. ' this.$eq = function (b) {',
  20130. ' return (this.D1 === b.D1) && (this.D2 === b.D2) && (this.D3 === b.D3) && (this.D4 === b.D4);',
  20131. ' };',
  20132. ' this.$assign = function (s) {',
  20133. ' this.D1 = s.D1;',
  20134. ' this.D2 = s.D2;',
  20135. ' this.D3 = s.D3;',
  20136. ' this.D4 = s.D4;',
  20137. ' return this;',
  20138. ' };',
  20139. '});',
  20140. 'this.DoConstGUIDIt = function (g) {',
  20141. '};',
  20142. 'this.DoDefGUID = function (g) {',
  20143. '};',
  20144. 'this.DoStr = function (s) {',
  20145. '};',
  20146. 'this.i = null;',
  20147. 'this.g = this.TGUID.$clone({',
  20148. ' D1: 0xD91C9AF4,',
  20149. ' D2: 0x3C93,',
  20150. ' D3: 0x420F,',
  20151. ' D4: [',
  20152. ' 0xA3,',
  20153. ' 0x03,',
  20154. ' 0xBF,',
  20155. ' 0x5B,',
  20156. ' 0xA8,',
  20157. ' 0x2B,',
  20158. ' 0xFD,',
  20159. ' 0x23',
  20160. ' ]',
  20161. '});',
  20162. 'this.s = "";',
  20163. '']),
  20164. LinesToStr([ // $mod.$main
  20165. '$mod.DoConstGUIDIt(rtl.getIntfGUIDR($mod.IUnknown));',
  20166. '$mod.DoDefGUID($mod.TGUID.$clone(rtl.getIntfGUIDR($mod.IUnknown)));',
  20167. '$mod.DoStr($mod.IUnknown.$guid);',
  20168. '$mod.DoConstGUIDIt(rtl.getIntfGUIDR($mod.i));',
  20169. '$mod.DoDefGUID($mod.TGUID.$clone(rtl.getIntfGUIDR($mod.i)));',
  20170. '$mod.DoStr($mod.i.$guid);',
  20171. '$mod.DoConstGUIDIt(rtl.strToGUIDR("{D91C9AF4-3c93-420f-A303-BF5BA82BFD23}"));',
  20172. '$mod.DoDefGUID(rtl.strToGUIDR("{D91C9AF4-3c93-420f-A303-BF5BA82BFD23}"));',
  20173. '$mod.DoStr(rtl.guidrToStr($mod.g));',
  20174. '$mod.g.$assign(rtl.getIntfGUIDR($mod.i));',
  20175. '$mod.g.$assign(rtl.getIntfGUIDR($mod.IUnknown));',
  20176. '$mod.g.$assign({',
  20177. ' D1: 0xD91C9AF4,',
  20178. ' D2: 0x3C93,',
  20179. ' D3: 0x420F,',
  20180. ' D4: [',
  20181. ' 0xA3,',
  20182. ' 0x03,',
  20183. ' 0xBF,',
  20184. ' 0x5B,',
  20185. ' 0xA8,',
  20186. ' 0x2B,',
  20187. ' 0xFD,',
  20188. ' 0x23',
  20189. ' ]',
  20190. '});',
  20191. '$mod.s = $mod.i.$guid;',
  20192. '$mod.s = $mod.IUnknown.$guid;',
  20193. '$mod.s = rtl.guidrToStr($mod.g);',
  20194. 'if ($mod.g.$eq(rtl.getIntfGUIDR($mod.i))) ;',
  20195. 'if ($mod.g.$eq(rtl.getIntfGUIDR($mod.i))) ;',
  20196. 'if ($mod.g.$eq(rtl.getIntfGUIDR($mod.IUnknown))) ;',
  20197. 'if ($mod.g.$eq(rtl.getIntfGUIDR($mod.IUnknown))) ;',
  20198. 'if ($mod.s === $mod.i.$guid) ;',
  20199. 'if ($mod.i.$guid === $mod.s) ;',
  20200. 'if ($mod.s === $mod.IUnknown.$guid) ;',
  20201. 'if ($mod.IUnknown.$guid === $mod.s) ;',
  20202. 'if ($mod.g.$eq(rtl.createTGUID($mod.s))) ;',
  20203. 'if ($mod.g.$eq(rtl.createTGUID($mod.s))) ;',
  20204. '']));
  20205. end;
  20206. procedure TTestModule.TestClassInterface_GUIDProperty;
  20207. begin
  20208. StartProgram(false);
  20209. Add([
  20210. '{$interfaces corba}',
  20211. 'type',
  20212. ' IUnknown = interface',
  20213. ' [''{f31db68f-3010-D355-4EBA-CDD4EF4A737C}'']',
  20214. ' end;',
  20215. ' TGUID = record D1, D2, D3, D4: word; end;',
  20216. ' TAliasGUID = TGUID;',
  20217. ' TGUIDString = type string;',
  20218. ' TAliasGUIDString = TGUIDString;',
  20219. ' TObject = class',
  20220. ' function GetG: TAliasGUID; virtual; abstract;',
  20221. ' procedure SetG(const Value: TAliasGUID); virtual; abstract;',
  20222. ' function GetS: TAliasGUIDString; virtual; abstract;',
  20223. ' procedure SetS(const Value: TAliasGUIDString); virtual; abstract;',
  20224. ' property g: TAliasGUID read GetG write SetG;',
  20225. ' property s: TAliasGUIDString read GetS write SetS;',
  20226. ' end;',
  20227. 'var o: TObject;',
  20228. 'begin',
  20229. ' o.g:=IUnknown;',
  20230. ' o.g:=''{D91C9AF4-3C93-420F-A303-bf5ba82bfd23}'';',
  20231. ' o.s:=IUnknown;',
  20232. ' o.s:=o.g;',
  20233. '']);
  20234. ConvertProgram;
  20235. CheckSource('TestClassInterface_GUIDProperty',
  20236. LinesToStr([ // statements
  20237. 'rtl.createInterface(this, "IUnknown", "{F31DB68F-3010-D355-4EBA-CDD4EF4A737C}", [], null);',
  20238. 'rtl.recNewT(this, "TGUID", function () {',
  20239. ' this.D1 = 0;',
  20240. ' this.D2 = 0;',
  20241. ' this.D3 = 0;',
  20242. ' this.D4 = 0;',
  20243. ' this.$eq = function (b) {',
  20244. ' return (this.D1 === b.D1) && (this.D2 === b.D2) && (this.D3 === b.D3) && (this.D4 === b.D4);',
  20245. ' };',
  20246. ' this.$assign = function (s) {',
  20247. ' this.D1 = s.D1;',
  20248. ' this.D2 = s.D2;',
  20249. ' this.D3 = s.D3;',
  20250. ' this.D4 = s.D4;',
  20251. ' return this;',
  20252. ' };',
  20253. '});',
  20254. 'rtl.createClass(this, "TObject", null, function () {',
  20255. ' this.$init = function () {',
  20256. ' };',
  20257. ' this.$final = function () {',
  20258. ' };',
  20259. '});',
  20260. 'this.o = null;',
  20261. '']),
  20262. LinesToStr([ // $mod.$main
  20263. '$mod.o.SetG(rtl.getIntfGUIDR($mod.IUnknown));',
  20264. '$mod.o.SetG({',
  20265. ' D1: 0xD91C9AF4,',
  20266. ' D2: 0x3C93,',
  20267. ' D3: 0x420F,',
  20268. ' D4: [',
  20269. ' 0xA3,',
  20270. ' 0x03,',
  20271. ' 0xBF,',
  20272. ' 0x5B,',
  20273. ' 0xA8,',
  20274. ' 0x2B,',
  20275. ' 0xFD,',
  20276. ' 0x23',
  20277. ' ]',
  20278. '});',
  20279. '$mod.o.SetS($mod.IUnknown.$guid);',
  20280. '$mod.o.SetS(rtl.guidrToStr($mod.o.GetG()));',
  20281. '']));
  20282. end;
  20283. procedure TTestModule.TestClassHelper_ClassVar;
  20284. begin
  20285. StartProgram(false);
  20286. Add([
  20287. 'type',
  20288. ' TObject = class',
  20289. ' end;',
  20290. ' THelper = class helper for TObject',
  20291. ' const',
  20292. ' One = 1;',
  20293. ' Two: word = 2;',
  20294. ' class var',
  20295. ' Glob: word;',
  20296. ' function Foo(w: word): word;',
  20297. ' class function Bar(w: word): word;',
  20298. ' end;',
  20299. 'function THelper.foo(w: word): word;',
  20300. 'begin',
  20301. ' Result:=w;',
  20302. ' Two:=One+w;',
  20303. ' Glob:=Glob;',
  20304. ' Result:=Self.Glob;',
  20305. ' Self.Glob:=Self.Glob;',
  20306. ' with Self do Glob:=Glob;',
  20307. 'end;',
  20308. 'class function THelper.bar(w: word): word;',
  20309. 'begin',
  20310. ' Result:=w;',
  20311. ' Two:=One;',
  20312. ' Glob:=Glob;',
  20313. ' Self.Glob:=Self.Glob;',
  20314. ' with Self do Glob:=Glob;',
  20315. 'end;',
  20316. 'var o: TObject;',
  20317. 'begin',
  20318. ' tobject.two:=tobject.one;',
  20319. ' tobject.Glob:=tobject.Glob;',
  20320. ' with tobject do begin',
  20321. ' two:=one;',
  20322. ' Glob:=Glob;',
  20323. ' end;',
  20324. ' o.two:=o.one;',
  20325. ' o.Glob:=o.Glob;',
  20326. ' with o do begin',
  20327. ' two:=one;',
  20328. ' Glob:=Glob;',
  20329. ' end;',
  20330. '']);
  20331. ConvertProgram;
  20332. CheckSource('TestClassHelper_ClassVar',
  20333. LinesToStr([ // statements
  20334. 'rtl.createClass(this, "TObject", null, function () {',
  20335. ' this.$init = function () {',
  20336. ' };',
  20337. ' this.$final = function () {',
  20338. ' };',
  20339. '});',
  20340. 'rtl.createHelper(this, "THelper", null, function () {',
  20341. ' this.One = 1;',
  20342. ' this.Two = 2;',
  20343. ' this.Glob = 0;',
  20344. ' this.Foo = function (w) {',
  20345. ' var Result = 0;',
  20346. ' Result = w;',
  20347. ' $mod.THelper.Two = 1 + w;',
  20348. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  20349. ' Result = $mod.THelper.Glob;',
  20350. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  20351. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  20352. ' return Result;',
  20353. ' };',
  20354. ' this.Bar = function (w) {',
  20355. ' var Result = 0;',
  20356. ' Result = w;',
  20357. ' $mod.THelper.Two = 1;',
  20358. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  20359. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  20360. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  20361. ' return Result;',
  20362. ' };',
  20363. '});',
  20364. 'this.o = null;',
  20365. '']),
  20366. LinesToStr([ // $mod.$main
  20367. '$mod.THelper.Two = 1;',
  20368. '$mod.THelper.Glob = $mod.THelper.Glob;',
  20369. 'var $with = $mod.TObject;',
  20370. '$mod.THelper.Two = 1;',
  20371. '$mod.THelper.Glob = $mod.THelper.Glob;',
  20372. '$mod.THelper.Two = 1;',
  20373. '$mod.THelper.Glob = $mod.THelper.Glob;',
  20374. 'var $with1 = $mod.o;',
  20375. '$mod.THelper.Two = 1;',
  20376. '$mod.THelper.Glob = $mod.THelper.Glob;',
  20377. '']));
  20378. end;
  20379. procedure TTestModule.TestClassHelper_Method_AccessInstanceFields;
  20380. begin
  20381. StartProgram(false);
  20382. Add([
  20383. 'type',
  20384. ' TObject = class',
  20385. ' FSize: word;',
  20386. ' property Size: word read FSize write FSize;',
  20387. ' end;',
  20388. ' THelper = class helper for TObject',
  20389. ' function Foo(w: word = 1): word;',
  20390. ' end;',
  20391. 'function THelper.foo(w: word): word;',
  20392. 'begin',
  20393. ' Result:=Size;',
  20394. ' Size:=Size+2;',
  20395. ' Self.Size:=Self.Size+3;',
  20396. ' FSize:=FSize+4;',
  20397. ' Self.FSize:=Self.FSize+5;',
  20398. ' with Self do begin',
  20399. ' Size:=Size+6;',
  20400. ' FSize:=FSize+7;',
  20401. ' FSize:=FSize+8;',
  20402. ' end;',
  20403. 'end;',
  20404. 'begin',
  20405. '']);
  20406. ConvertProgram;
  20407. CheckSource('TestClassHelper_Method_AccessInstanceFields',
  20408. LinesToStr([ // statements
  20409. 'rtl.createClass(this, "TObject", null, function () {',
  20410. ' this.$init = function () {',
  20411. ' this.FSize = 0;',
  20412. ' };',
  20413. ' this.$final = function () {',
  20414. ' };',
  20415. '});',
  20416. 'rtl.createHelper(this, "THelper", null, function () {',
  20417. ' this.Foo = function (w) {',
  20418. ' var Result = 0;',
  20419. ' Result = this.FSize;',
  20420. ' this.FSize = this.FSize + 2;',
  20421. ' this.FSize = this.FSize + 3;',
  20422. ' this.FSize = this.FSize + 4;',
  20423. ' this.FSize = this.FSize + 5;',
  20424. ' this.FSize = this.FSize + 6;',
  20425. ' this.FSize = this.FSize + 7;',
  20426. ' this.FSize = this.FSize + 8;',
  20427. ' return Result;',
  20428. ' };',
  20429. '});',
  20430. '']),
  20431. LinesToStr([ // $mod.$main
  20432. '']));
  20433. end;
  20434. procedure TTestModule.TestClassHelper_Method_Call;
  20435. begin
  20436. StartProgram(false);
  20437. Add([
  20438. 'type',
  20439. ' TObject = class',
  20440. ' procedure Run(w: word = 10);',
  20441. ' end;',
  20442. ' THelper = class helper for TObject',
  20443. ' function Foo(w: word = 1): word;',
  20444. ' end;',
  20445. 'procedure TObject.Run(w: word);',
  20446. 'var o: TObject;',
  20447. 'begin',
  20448. ' Foo;',
  20449. ' Foo();',
  20450. ' Foo(2);',
  20451. ' Self.Foo;',
  20452. ' Self.Foo();',
  20453. ' Self.Foo(3);',
  20454. ' with Self do begin',
  20455. ' Foo;',
  20456. ' Foo();',
  20457. ' Foo(4);',
  20458. ' end;',
  20459. ' with o do Foo(5);',
  20460. 'end;',
  20461. 'function THelper.foo(w: word): word;',
  20462. 'begin',
  20463. ' Run;',
  20464. ' Run();',
  20465. ' Run(11);',
  20466. ' Foo;',
  20467. ' Foo();',
  20468. ' Foo(12);',
  20469. ' Self.Foo;',
  20470. ' Self.Foo();',
  20471. ' Self.Foo(13);',
  20472. ' with Self do begin',
  20473. ' Foo;',
  20474. ' Foo();',
  20475. ' Foo(14);',
  20476. ' end;',
  20477. 'end;',
  20478. 'var Obj: TObject;',
  20479. 'begin',
  20480. ' obj.Foo;',
  20481. ' obj.Foo();',
  20482. ' obj.Foo(21);',
  20483. ' with obj do begin',
  20484. ' Foo;',
  20485. ' Foo();',
  20486. ' Foo(22);',
  20487. ' end;',
  20488. '']);
  20489. ConvertProgram;
  20490. CheckSource('TestClassHelper_Method_Call',
  20491. LinesToStr([ // statements
  20492. 'rtl.createClass(this, "TObject", null, function () {',
  20493. ' this.$init = function () {',
  20494. ' };',
  20495. ' this.$final = function () {',
  20496. ' };',
  20497. ' this.Run = function (w) {',
  20498. ' var o = null;',
  20499. ' $mod.THelper.Foo.call(this, 1);',
  20500. ' $mod.THelper.Foo.call(this, 1);',
  20501. ' $mod.THelper.Foo.call(this, 2);',
  20502. ' $mod.THelper.Foo.call(this, 1);',
  20503. ' $mod.THelper.Foo.call(this, 1);',
  20504. ' $mod.THelper.Foo.call(this, 3);',
  20505. ' $mod.THelper.Foo.call(this, 1);',
  20506. ' $mod.THelper.Foo.call(this, 1);',
  20507. ' $mod.THelper.Foo.call(this, 4);',
  20508. ' $mod.THelper.Foo.call(o, 5);',
  20509. ' };',
  20510. '});',
  20511. 'rtl.createHelper(this, "THelper", null, function () {',
  20512. ' this.Foo = function (w) {',
  20513. ' var Result = 0;',
  20514. ' this.Run(10);',
  20515. ' this.Run(10);',
  20516. ' this.Run(11);',
  20517. ' $mod.THelper.Foo.call(this, 1);',
  20518. ' $mod.THelper.Foo.call(this, 1);',
  20519. ' $mod.THelper.Foo.call(this, 12);',
  20520. ' $mod.THelper.Foo.call(this, 1);',
  20521. ' $mod.THelper.Foo.call(this, 1);',
  20522. ' $mod.THelper.Foo.call(this, 13);',
  20523. ' $mod.THelper.Foo.call(this, 1);',
  20524. ' $mod.THelper.Foo.call(this, 1);',
  20525. ' $mod.THelper.Foo.call(this, 14);',
  20526. ' return Result;',
  20527. ' };',
  20528. '});',
  20529. 'this.Obj = null;',
  20530. '']),
  20531. LinesToStr([ // $mod.$main
  20532. '$mod.THelper.Foo.call($mod.Obj, 1);',
  20533. '$mod.THelper.Foo.call($mod.Obj, 1);',
  20534. '$mod.THelper.Foo.call($mod.Obj, 21);',
  20535. 'var $with = $mod.Obj;',
  20536. '$mod.THelper.Foo.call($with, 1);',
  20537. '$mod.THelper.Foo.call($with, 1);',
  20538. '$mod.THelper.Foo.call($with, 22);',
  20539. '']));
  20540. end;
  20541. procedure TTestModule.TestClassHelper_Method_Nested_Call;
  20542. begin
  20543. StartProgram(false);
  20544. Add([
  20545. 'type',
  20546. ' TObject = class',
  20547. ' procedure Run(w: word = 10);',
  20548. ' end;',
  20549. ' THelper = class helper for TObject',
  20550. ' function Foo(w: word = 1): word;',
  20551. ' end;',
  20552. 'procedure TObject.Run(w: word);',
  20553. ' procedure Sub(Self: TObject);',
  20554. ' begin',
  20555. ' Foo;',
  20556. ' Foo();',
  20557. ' Self.Foo;',
  20558. ' Self.Foo();',
  20559. ' with Self do begin',
  20560. ' Foo;',
  20561. ' Foo();',
  20562. ' end;',
  20563. ' end;',
  20564. 'begin',
  20565. 'end;',
  20566. 'function THelper.foo(w: word): word;',
  20567. ' procedure Sub(Self: TObject);',
  20568. ' begin',
  20569. ' Run;',
  20570. ' Run();',
  20571. ' Foo;',
  20572. ' Foo();',
  20573. ' Self.Foo;',
  20574. ' Self.Foo();',
  20575. ' with Self do begin',
  20576. ' Foo;',
  20577. ' Foo();',
  20578. ' end;',
  20579. ' end;',
  20580. 'begin',
  20581. 'end;',
  20582. 'begin',
  20583. '']);
  20584. ConvertProgram;
  20585. CheckSource('TestClassHelper_Method_Nested_Call',
  20586. LinesToStr([ // statements
  20587. 'rtl.createClass(this, "TObject", null, function () {',
  20588. ' this.$init = function () {',
  20589. ' };',
  20590. ' this.$final = function () {',
  20591. ' };',
  20592. ' this.Run = function (w) {',
  20593. ' var $Self = this;',
  20594. ' function Sub(Self) {',
  20595. ' $mod.THelper.Foo.call($Self, 1);',
  20596. ' $mod.THelper.Foo.call($Self, 1);',
  20597. ' $mod.THelper.Foo.call(Self, 1);',
  20598. ' $mod.THelper.Foo.call(Self, 1);',
  20599. ' $mod.THelper.Foo.call(Self, 1);',
  20600. ' $mod.THelper.Foo.call(Self, 1);',
  20601. ' };',
  20602. ' };',
  20603. '});',
  20604. 'rtl.createHelper(this, "THelper", null, function () {',
  20605. ' this.Foo = function (w) {',
  20606. ' var $Self = this;',
  20607. ' var Result = 0;',
  20608. ' function Sub(Self) {',
  20609. ' $Self.Run(10);',
  20610. ' $Self.Run(10);',
  20611. ' $mod.THelper.Foo.call($Self, 1);',
  20612. ' $mod.THelper.Foo.call($Self, 1);',
  20613. ' $mod.THelper.Foo.call(Self, 1);',
  20614. ' $mod.THelper.Foo.call(Self, 1);',
  20615. ' $mod.THelper.Foo.call(Self, 1);',
  20616. ' $mod.THelper.Foo.call(Self, 1);',
  20617. ' };',
  20618. ' return Result;',
  20619. ' };',
  20620. '});',
  20621. '']),
  20622. LinesToStr([ // $mod.$main
  20623. '']));
  20624. end;
  20625. procedure TTestModule.TestClassHelper_ClassMethod_Call;
  20626. begin
  20627. StartProgram(false);
  20628. Add([
  20629. 'type',
  20630. ' TObject = class',
  20631. ' class procedure Run(w: word = 10);',
  20632. ' end;',
  20633. ' THelper = class helper for TObject',
  20634. ' class function Foo(w: word = 1): word;',
  20635. ' end;',
  20636. 'class procedure TObject.Run(w: word);',
  20637. 'begin',
  20638. ' Foo;',
  20639. ' Foo();',
  20640. ' Self.Foo;',
  20641. ' Self.Foo();',
  20642. ' with Self do begin',
  20643. ' Foo;',
  20644. ' Foo();',
  20645. ' end;',
  20646. 'end;',
  20647. 'class function THelper.foo(w: word): word;',
  20648. 'begin',
  20649. ' Run;',
  20650. ' Run();',
  20651. ' Foo;',
  20652. ' Foo();',
  20653. ' Self.Foo;',
  20654. ' Self.Foo();',
  20655. ' with Self do begin',
  20656. ' Foo;',
  20657. ' Foo();',
  20658. ' end;',
  20659. 'end;',
  20660. 'var',
  20661. ' Obj: TObject;',
  20662. 'begin',
  20663. ' obj.Foo;',
  20664. ' obj.Foo();',
  20665. ' with obj do begin',
  20666. ' Foo;',
  20667. ' Foo();',
  20668. ' end;',
  20669. ' tobject.Foo;',
  20670. ' tobject.Foo();',
  20671. ' with tobject do begin',
  20672. ' Foo;',
  20673. ' Foo();',
  20674. ' end;',
  20675. '']);
  20676. ConvertProgram;
  20677. CheckSource('TestClassHelper_ClassMethod_Call',
  20678. LinesToStr([ // statements
  20679. 'rtl.createClass(this, "TObject", null, function () {',
  20680. ' this.$init = function () {',
  20681. ' };',
  20682. ' this.$final = function () {',
  20683. ' };',
  20684. ' this.Run = function (w) {',
  20685. ' $mod.THelper.Foo.call(this, 1);',
  20686. ' $mod.THelper.Foo.call(this, 1);',
  20687. ' $mod.THelper.Foo.call(this, 1);',
  20688. ' $mod.THelper.Foo.call(this, 1);',
  20689. ' $mod.THelper.Foo.call(this, 1);',
  20690. ' $mod.THelper.Foo.call(this, 1);',
  20691. ' };',
  20692. '});',
  20693. 'rtl.createHelper(this, "THelper", null, function () {',
  20694. ' this.Foo = function (w) {',
  20695. ' var Result = 0;',
  20696. ' this.Run(10);',
  20697. ' this.Run(10);',
  20698. ' $mod.THelper.Foo.call(this, 1);',
  20699. ' $mod.THelper.Foo.call(this, 1);',
  20700. ' $mod.THelper.Foo.call(this, 1);',
  20701. ' $mod.THelper.Foo.call(this, 1);',
  20702. ' $mod.THelper.Foo.call(this, 1);',
  20703. ' $mod.THelper.Foo.call(this, 1);',
  20704. ' return Result;',
  20705. ' };',
  20706. '});',
  20707. 'this.Obj = null;',
  20708. '']),
  20709. LinesToStr([ // $mod.$main
  20710. '$mod.THelper.Foo.call($mod.Obj.$class, 1);',
  20711. '$mod.THelper.Foo.call($mod.Obj.$class, 1);',
  20712. 'var $with = $mod.Obj;',
  20713. '$mod.THelper.Foo.call($with.$class, 1);',
  20714. '$mod.THelper.Foo.call($with.$class, 1);',
  20715. '$mod.THelper.Foo.call($mod.TObject, 1);',
  20716. '$mod.THelper.Foo.call($mod.TObject, 1);',
  20717. 'var $with1 = $mod.TObject;',
  20718. '$mod.THelper.Foo.call($mod.TObject, 1);',
  20719. '$mod.THelper.Foo.call($mod.TObject, 1);',
  20720. '']));
  20721. end;
  20722. procedure TTestModule.TestClassHelper_ClassOf;
  20723. begin
  20724. StartProgram(false);
  20725. Add([
  20726. 'type',
  20727. ' TObject = class',
  20728. ' end;',
  20729. ' TClass = class of TObject;',
  20730. ' THelper = class helper for TObject',
  20731. ' class function Foo(w: word = 1): word;',
  20732. ' end;',
  20733. 'class function THelper.foo(w: word): word;',
  20734. 'begin',
  20735. 'end;',
  20736. 'var',
  20737. ' c: TClass;',
  20738. 'begin',
  20739. ' c.Foo;',
  20740. ' c.Foo();',
  20741. ' with c do begin',
  20742. ' Foo;',
  20743. ' Foo();',
  20744. ' end;',
  20745. '']);
  20746. ConvertProgram;
  20747. CheckSource('TestClassHelper_ClassOf',
  20748. LinesToStr([ // statements
  20749. 'rtl.createClass(this, "TObject", null, function () {',
  20750. ' this.$init = function () {',
  20751. ' };',
  20752. ' this.$final = function () {',
  20753. ' };',
  20754. '});',
  20755. 'rtl.createHelper(this, "THelper", null, function () {',
  20756. ' this.Foo = function (w) {',
  20757. ' var Result = 0;',
  20758. ' return Result;',
  20759. ' };',
  20760. '});',
  20761. 'this.c = null;',
  20762. '']),
  20763. LinesToStr([ // $mod.$main
  20764. '$mod.THelper.Foo.call($mod.c, 1);',
  20765. '$mod.THelper.Foo.call($mod.c, 1);',
  20766. 'var $with = $mod.c;',
  20767. '$mod.THelper.Foo.call($with, 1);',
  20768. '$mod.THelper.Foo.call($with, 1);',
  20769. '']));
  20770. end;
  20771. procedure TTestModule.TestClassHelper_MethodRefObjFPC;
  20772. begin
  20773. StartProgram(false);
  20774. Add([
  20775. '{$mode objfpc}',
  20776. 'type',
  20777. ' TObject = class',
  20778. ' procedure DoIt;',
  20779. ' end;',
  20780. ' THelper = class helper for TObject',
  20781. ' procedure Fly(w: word = 1);',
  20782. ' class procedure Glide(w: word = 1);',
  20783. ' class procedure Run(w: word = 1); static;',
  20784. ' end;',
  20785. ' TFly = procedure(w: word) of object;',
  20786. ' TGlide = TFly;',
  20787. ' TRun = procedure(w: word);',
  20788. 'var',
  20789. ' f: TFly;',
  20790. ' g: TGlide;',
  20791. ' r: TRun;',
  20792. 'procedure TObject.DoIt;',
  20793. 'begin',
  20794. ' f:=@fly;',
  20795. ' g:=@glide;',
  20796. ' r:=@run;',
  20797. ' f:[email protected];',
  20798. ' g:[email protected];',
  20799. ' r:[email protected];',
  20800. ' with self do begin',
  20801. ' f:=@fly;',
  20802. ' g:=@glide;',
  20803. ' r:=@run;',
  20804. ' end;',
  20805. 'end;',
  20806. 'procedure THelper.fly(w: word);',
  20807. 'begin',
  20808. ' f:=@fly;',
  20809. ' g:=@glide;',
  20810. ' r:=@run;',
  20811. 'end;',
  20812. 'class procedure THelper.glide(w: word);',
  20813. 'begin',
  20814. ' g:=@glide;',
  20815. ' r:=@run;',
  20816. 'end;',
  20817. 'class procedure THelper.run(w: word);',
  20818. 'begin',
  20819. ' g:=@glide;',
  20820. ' r:=@run;',
  20821. 'end;',
  20822. 'var',
  20823. ' Obj: TObject;',
  20824. 'begin',
  20825. ' f:[email protected];',
  20826. ' g:[email protected];',
  20827. ' r:[email protected];',
  20828. ' with obj do begin',
  20829. ' f:=@fly;',
  20830. ' g:=@glide;',
  20831. ' r:=@run;',
  20832. ' end;',
  20833. ' g:[email protected];',
  20834. ' r:[email protected];',
  20835. ' with tobject do begin',
  20836. ' g:=@glide;',
  20837. ' r:=@run;',
  20838. ' end;',
  20839. '']);
  20840. ConvertProgram;
  20841. CheckSource('TestClassHelper_MethodRefObjFPC',
  20842. LinesToStr([ // statements
  20843. 'rtl.createClass(this, "TObject", null, function () {',
  20844. ' this.$init = function () {',
  20845. ' };',
  20846. ' this.$final = function () {',
  20847. ' };',
  20848. ' this.DoIt = function () {',
  20849. ' $mod.f = rtl.createCallback(this, $mod.THelper.Fly);',
  20850. ' $mod.g = rtl.createCallback(this.$class, $mod.THelper.Glide);',
  20851. ' $mod.r = $mod.THelper.Run;',
  20852. ' $mod.f = rtl.createCallback(this, $mod.THelper.Fly);',
  20853. ' $mod.g = rtl.createCallback(this.$class, $mod.THelper.Glide);',
  20854. ' $mod.r = $mod.THelper.Run;',
  20855. ' $mod.f = rtl.createCallback(this, $mod.THelper.Fly);',
  20856. ' $mod.g = rtl.createCallback(this.$class, $mod.THelper.Glide);',
  20857. ' $mod.r = $mod.THelper.Run;',
  20858. ' };',
  20859. '});',
  20860. 'rtl.createHelper(this, "THelper", null, function () {',
  20861. ' this.Fly = function (w) {',
  20862. ' $mod.f = rtl.createCallback(this, $mod.THelper.Fly);',
  20863. ' $mod.g = rtl.createCallback(this.$class, $mod.THelper.Glide);',
  20864. ' $mod.r = $mod.THelper.Run;',
  20865. ' };',
  20866. ' this.Glide = function (w) {',
  20867. ' $mod.g = rtl.createCallback(this, $mod.THelper.Glide);',
  20868. ' $mod.r = $mod.THelper.Run;',
  20869. ' };',
  20870. ' this.Run = function (w) {',
  20871. ' $mod.g = rtl.createCallback($mod.THelper, $mod.THelper.Glide);',
  20872. ' $mod.r = $mod.THelper.Run;',
  20873. ' };',
  20874. '});',
  20875. 'this.f = null;',
  20876. 'this.g = null;',
  20877. 'this.r = null;',
  20878. 'this.Obj = null;',
  20879. '']),
  20880. LinesToStr([ // $mod.$main
  20881. '$mod.f = rtl.createCallback($mod.Obj, $mod.THelper.Fly);',
  20882. '$mod.g = rtl.createCallback($mod.Obj.$class, $mod.THelper.Glide);',
  20883. '$mod.r = $mod.THelper.Run;',
  20884. 'var $with = $mod.Obj;',
  20885. '$mod.f = rtl.createCallback($with, $mod.THelper.Fly);',
  20886. '$mod.g = rtl.createCallback($with.$class, $mod.THelper.Glide);',
  20887. '$mod.r = $mod.THelper.Run;',
  20888. '$mod.g = rtl.createCallback($mod.TObject, $mod.THelper.Glide);',
  20889. '$mod.r = $mod.THelper.Run;',
  20890. 'var $with1 = $mod.TObject;',
  20891. '$mod.g = rtl.createCallback($with1, $mod.THelper.Glide);',
  20892. '$mod.r = $mod.THelper.Run;',
  20893. '']));
  20894. end;
  20895. procedure TTestModule.TestClassHelper_Constructor;
  20896. begin
  20897. StartProgram(false);
  20898. Add([
  20899. 'type',
  20900. ' TObject = class',
  20901. ' constructor Create;',
  20902. ' end;',
  20903. ' TClass = class of TObject;',
  20904. ' THelper = class helper for TObject',
  20905. ' constructor NewHlp(w: word);',
  20906. ' end;',
  20907. 'var',
  20908. ' obj: TObject;',
  20909. ' c: TClass;',
  20910. 'constructor TObject.Create;',
  20911. 'begin',
  20912. ' NewHlp(2);', // normal call
  20913. ' tobject.NewHlp(3);', // new instance
  20914. ' c.newhlp(4);', // new instance
  20915. 'end;',
  20916. 'constructor THelper.NewHlp(w: word);',
  20917. 'begin',
  20918. ' create;', // normal call
  20919. ' tobject.create;', // new instance
  20920. ' NewHlp(2);', // normal call
  20921. ' tobject.NewHlp(3);', // new instance
  20922. ' c.newhlp(4);', // new instance
  20923. 'end;',
  20924. 'begin',
  20925. ' obj.newhlp(2);', // normal call
  20926. ' with Obj do newhlp(12);', // normal call
  20927. ' tobject.newhlp(3);', // new instance
  20928. ' with tobject do newhlp(13);', // new instance
  20929. ' c.newhlp(4);', // new instance
  20930. ' with c do newhlp(14);', // new instance
  20931. '']);
  20932. ConvertProgram;
  20933. CheckSource('TestClassHelper_Constructor',
  20934. LinesToStr([ // statements
  20935. 'rtl.createClass(this, "TObject", null, function () {',
  20936. ' this.$init = function () {',
  20937. ' };',
  20938. ' this.$final = function () {',
  20939. ' };',
  20940. ' this.Create = function () {',
  20941. ' $mod.THelper.NewHlp.call(this, 2);',
  20942. ' $mod.TObject.$create($mod.THelper.NewHlp, [3]);',
  20943. ' $mod.c.$create($mod.THelper.NewHlp, [4]);',
  20944. ' return this;',
  20945. ' };',
  20946. '});',
  20947. 'rtl.createHelper(this, "THelper", null, function () {',
  20948. ' this.NewHlp = function (w) {',
  20949. ' this.Create();',
  20950. ' $mod.TObject.$create("Create");',
  20951. ' $mod.THelper.NewHlp.call(this, 2);',
  20952. ' $mod.TObject.$create($mod.THelper.NewHlp, [3]);',
  20953. ' $mod.c.$create($mod.THelper.NewHlp, [4]);',
  20954. ' return this;',
  20955. ' };',
  20956. '});',
  20957. 'this.obj = null;',
  20958. 'this.c = null;',
  20959. '']),
  20960. LinesToStr([ // $mod.$main
  20961. '$mod.THelper.NewHlp.call($mod.obj, 2);',
  20962. 'var $with = $mod.obj;',
  20963. '$mod.THelper.NewHlp.call($with, 12);',
  20964. '$mod.TObject.$create($mod.THelper.NewHlp, [3]);',
  20965. 'var $with1 = $mod.TObject;',
  20966. '$with1.$create($mod.THelper.NewHlp, [13]);',
  20967. '$mod.c.$create($mod.THelper.NewHlp, [4]);',
  20968. 'var $with2 = $mod.c;',
  20969. '$with2.$create($mod.THelper.NewHlp, [14]);',
  20970. '']));
  20971. end;
  20972. procedure TTestModule.TestClassHelper_InheritedObjFPC;
  20973. begin
  20974. StartProgram(false);
  20975. Add([
  20976. 'type',
  20977. ' TObject = class',
  20978. ' procedure Fly;',
  20979. ' end;',
  20980. ' TObjHelper = class helper for TObject',
  20981. ' procedure Fly;',
  20982. ' end;',
  20983. ' TBird = class',
  20984. ' procedure Fly;',
  20985. ' end;',
  20986. ' TBirdHelper = class helper for TBird',
  20987. ' procedure Fly;',
  20988. ' procedure Walk(w: word);',
  20989. ' end;',
  20990. ' TEagleHelper = class helper(TBirdHelper) for TBird',
  20991. ' procedure Fly;',
  20992. ' procedure Walk(w: word);',
  20993. ' end;',
  20994. 'procedure Tobject.fly;',
  20995. 'begin',
  20996. ' inherited;', // ignore
  20997. 'end;',
  20998. 'procedure Tobjhelper.fly;',
  20999. 'begin',
  21000. ' {@TObject_Fly}inherited;',
  21001. ' inherited {@TObject_Fly}Fly;',
  21002. 'end;',
  21003. 'procedure Tbird.fly;',
  21004. 'begin',
  21005. ' {@TObjHelper_Fly}inherited;',
  21006. ' inherited {@TObjHelper_Fly}Fly;',
  21007. 'end;',
  21008. 'procedure Tbirdhelper.fly;',
  21009. 'begin',
  21010. ' {@TBird_Fly}inherited;',
  21011. ' inherited {@TBird_Fly}Fly;',
  21012. 'end;',
  21013. 'procedure Tbirdhelper.walk(w: word);',
  21014. 'begin',
  21015. 'end;',
  21016. 'procedure teagleHelper.fly;',
  21017. 'begin',
  21018. ' {@TBird_Fly}inherited;',
  21019. ' inherited {@TBird_Fly}Fly;',
  21020. 'end;',
  21021. 'procedure teagleHelper.walk(w: word);',
  21022. 'begin',
  21023. ' {@TBirdHelper_Walk}inherited;',
  21024. ' inherited {@TBirdHelper_Walk}Walk(3);',
  21025. 'end;',
  21026. 'begin',
  21027. '']);
  21028. ConvertProgram;
  21029. CheckSource('TestClassHelper_InheritedObjFPC',
  21030. LinesToStr([ // statements
  21031. 'rtl.createClass(this, "TObject", null, function () {',
  21032. ' this.$init = function () {',
  21033. ' };',
  21034. ' this.$final = function () {',
  21035. ' };',
  21036. ' this.Fly = function () {',
  21037. ' };',
  21038. '});',
  21039. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21040. ' this.Fly = function () {',
  21041. ' $mod.TObject.Fly.call(this);',
  21042. ' $mod.TObject.Fly.call(this);',
  21043. ' };',
  21044. '});',
  21045. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21046. ' this.Fly$1 = function () {',
  21047. ' $mod.TObjHelper.Fly.call(this);',
  21048. ' $mod.TObjHelper.Fly.call(this);',
  21049. ' };',
  21050. '});',
  21051. 'rtl.createHelper(this, "TBirdHelper", null, function () {',
  21052. ' this.Fly = function () {',
  21053. ' $mod.TBird.Fly$1.call(this);',
  21054. ' $mod.TBird.Fly$1.call(this);',
  21055. ' };',
  21056. ' this.Walk = function (w) {',
  21057. ' };',
  21058. '});',
  21059. 'rtl.createHelper(this, "TEagleHelper", this.TBirdHelper, function () {',
  21060. ' this.Fly$1 = function () {',
  21061. ' $mod.TBird.Fly$1.call(this);',
  21062. ' $mod.TBird.Fly$1.call(this);',
  21063. ' };',
  21064. ' this.Walk$1 = function (w) {',
  21065. ' $mod.TBirdHelper.Walk.apply(this, arguments);',
  21066. ' $mod.TBirdHelper.Walk.call(this, 3);',
  21067. ' };',
  21068. '});',
  21069. '']),
  21070. LinesToStr([ // $mod.$main
  21071. '']));
  21072. end;
  21073. procedure TTestModule.TestClassHelper_Property;
  21074. begin
  21075. StartProgram(false);
  21076. Add([
  21077. 'type',
  21078. ' TObject = class',
  21079. ' FSize: word;',
  21080. ' function GetSpeed: word;',
  21081. ' procedure SetSpeed(Value: word);',
  21082. ' end;',
  21083. ' TObjHelper = class helper for TObject',
  21084. ' function GetLeft: word;',
  21085. ' procedure SetLeft(Value: word);',
  21086. ' property Size: word read FSize write FSize;',
  21087. ' property Speed: word read GetSpeed write SetSpeed;',
  21088. ' property Left: word read GetLeft write SetLeft;',
  21089. ' end;',
  21090. ' TBird = class',
  21091. ' property NotRight: word read GetLeft write SetLeft;',
  21092. ' procedure DoIt;',
  21093. ' end;',
  21094. 'var',
  21095. ' b: TBird;',
  21096. 'function Tobject.GetSpeed: word;',
  21097. 'begin',
  21098. ' Size:=Size+11;',
  21099. ' Speed:=Speed+12;',
  21100. ' Result:=Left+13;',
  21101. ' Left:=13;',
  21102. ' Left:=Left+13;',
  21103. ' Self.Size:=Self.Size+21;',
  21104. ' Self.Speed:=Self.Speed+22;',
  21105. ' Self.Left:=Self.Left+23;',
  21106. ' with Self do begin',
  21107. ' Size:=Size+31;',
  21108. ' Speed:=Speed+32;',
  21109. ' Left:=Left+33;',
  21110. ' end;',
  21111. 'end;',
  21112. 'procedure Tobject.SetSpeed(Value: word);',
  21113. 'begin',
  21114. 'end;',
  21115. 'function TObjHelper.GetLeft: word;',
  21116. 'begin',
  21117. ' Size:=Size+11;',
  21118. ' Speed:=Speed+12;',
  21119. ' Left:=Left+13;',
  21120. ' Self.Size:=Self.Size+21;',
  21121. ' Self.Speed:=Self.Speed+22;',
  21122. ' Self.Left:=Self.Left+23;',
  21123. ' with Self do begin',
  21124. ' Size:=Size+31;',
  21125. ' Speed:=Speed+32;',
  21126. ' Left:=Left+33;',
  21127. ' end;',
  21128. 'end;',
  21129. 'procedure TObjHelper.SetLeft(Value: word);',
  21130. 'begin',
  21131. 'end;',
  21132. 'procedure TBird.DoIt;',
  21133. 'begin',
  21134. ' NotRight:=NotRight+11;',
  21135. ' Self.NotRight:=Self.NotRight+21;',
  21136. ' with Self do begin',
  21137. ' NotRight:=NotRight+31;',
  21138. ' end;',
  21139. 'end;',
  21140. 'begin',
  21141. ' b.Size:=b.Size+11;',
  21142. ' b.Speed:=b.Speed+12;',
  21143. ' b.Left:=b.Left+13;',
  21144. ' b.NotRight:=b.NotRight+14;',
  21145. ' with b do begin',
  21146. ' Size:=Size+31;',
  21147. ' Speed:=Speed+32;',
  21148. ' Left:=Left+33;',
  21149. ' NotRight:=NotRight+34;',
  21150. ' end;',
  21151. '']);
  21152. ConvertProgram;
  21153. CheckSource('TestClassHelper_Property',
  21154. LinesToStr([ // statements
  21155. 'rtl.createClass(this, "TObject", null, function () {',
  21156. ' this.$init = function () {',
  21157. ' this.FSize = 0;',
  21158. ' };',
  21159. ' this.$final = function () {',
  21160. ' };',
  21161. ' this.GetSpeed = function () {',
  21162. ' var Result = 0;',
  21163. ' this.FSize = this.FSize + 11;',
  21164. ' this.SetSpeed(this.GetSpeed() + 12);',
  21165. ' Result = $mod.TObjHelper.GetLeft.call(this) + 13;',
  21166. ' $mod.TObjHelper.SetLeft.call(this, 13);',
  21167. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 13);',
  21168. ' this.FSize = this.FSize + 21;',
  21169. ' this.SetSpeed(this.GetSpeed() + 22);',
  21170. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 23);',
  21171. ' this.FSize = this.FSize + 31;',
  21172. ' this.SetSpeed(this.GetSpeed() + 32);',
  21173. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 33);',
  21174. ' return Result;',
  21175. ' };',
  21176. ' this.SetSpeed = function (Value) {',
  21177. ' };',
  21178. '});',
  21179. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21180. ' this.GetLeft = function () {',
  21181. ' var Result = 0;',
  21182. ' this.FSize = this.FSize + 11;',
  21183. ' this.SetSpeed(this.GetSpeed() + 12);',
  21184. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 13);',
  21185. ' this.FSize = this.FSize + 21;',
  21186. ' this.SetSpeed(this.GetSpeed() + 22);',
  21187. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 23);',
  21188. ' this.FSize = this.FSize + 31;',
  21189. ' this.SetSpeed(this.GetSpeed() + 32);',
  21190. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 33);',
  21191. ' return Result;',
  21192. ' };',
  21193. ' this.SetLeft = function (Value) {',
  21194. ' };',
  21195. '});',
  21196. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21197. ' this.DoIt = function () {',
  21198. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 11);',
  21199. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 21);',
  21200. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 31);',
  21201. ' };',
  21202. '});',
  21203. 'this.b = null;',
  21204. '']),
  21205. LinesToStr([ // $mod.$main
  21206. '$mod.b.FSize = $mod.b.FSize + 11;',
  21207. '$mod.b.SetSpeed($mod.b.GetSpeed() + 12);',
  21208. '$mod.TObjHelper.SetLeft.call($mod.b, $mod.TObjHelper.GetLeft.call($mod.b) + 13);',
  21209. '$mod.TObjHelper.SetLeft.call($mod.b, $mod.TObjHelper.GetLeft.call($mod.b) + 14);',
  21210. 'var $with = $mod.b;',
  21211. '$with.FSize = $with.FSize + 31;',
  21212. '$with.SetSpeed($with.GetSpeed() + 32);',
  21213. '$mod.TObjHelper.SetLeft.call($with, $mod.TObjHelper.GetLeft.call($with) + 33);',
  21214. '$mod.TObjHelper.SetLeft.call($with, $mod.TObjHelper.GetLeft.call($with) + 34);',
  21215. '']));
  21216. end;
  21217. procedure TTestModule.TestClassHelper_Property_Array;
  21218. begin
  21219. StartProgram(false);
  21220. Add([
  21221. 'type',
  21222. ' TObject = class',
  21223. ' function GetSpeed(Index: boolean): word;',
  21224. ' procedure SetSpeed(Index: boolean; Value: word);',
  21225. ' end;',
  21226. ' TObjHelper = class helper for TObject',
  21227. ' function GetSize(Index: boolean): word;',
  21228. ' procedure SetSize(Index: boolean; Value: word);',
  21229. ' property Size[Index: boolean]: word read GetSize write SetSize;',
  21230. ' property Speed[Index: boolean]: word read GetSpeed write SetSpeed;',
  21231. ' end;',
  21232. ' TBird = class',
  21233. ' property Items[Index: boolean]: word read GetSize write SetSize;',
  21234. ' procedure DoIt;',
  21235. ' end;',
  21236. 'var',
  21237. ' b: TBird;',
  21238. 'function Tobject.GetSpeed(Index: boolean): word;',
  21239. 'begin',
  21240. ' Result:=Size[false];',
  21241. ' Size[true]:=Size[false]+11;',
  21242. ' Speed[true]:=Speed[false]+12;',
  21243. ' Self.Size[true]:=Self.Size[false]+21;',
  21244. ' Self.Speed[true]:=Self.Speed[false]+22;',
  21245. ' with Self do begin',
  21246. ' Size[true]:=Size[false]+31;',
  21247. ' Speed[true]:=Speed[false]+32;',
  21248. ' end;',
  21249. 'end;',
  21250. 'procedure Tobject.SetSpeed(Index: boolean; Value: word);',
  21251. 'begin',
  21252. 'end;',
  21253. 'function TObjHelper.GetSize(Index: boolean): word;',
  21254. 'begin',
  21255. ' Size[true]:=Size[false]+11;',
  21256. ' Speed[true]:=Speed[false]+12;',
  21257. ' Self.Size[true]:=Self.Size[false]+21;',
  21258. ' Self.Speed[true]:=Self.Speed[false]+22;',
  21259. ' with Self do begin',
  21260. ' Size[true]:=Size[false]+31;',
  21261. ' Speed[true]:=Speed[false]+32;',
  21262. ' end;',
  21263. 'end;',
  21264. 'procedure TObjHelper.SetSize(Index: boolean; Value: word);',
  21265. 'begin',
  21266. 'end;',
  21267. 'procedure TBird.DoIt;',
  21268. 'begin',
  21269. ' Items[true]:=Items[false]+11;',
  21270. ' Self.Items[true]:=Self.Items[false]+21;',
  21271. ' with Self do Items[true]:=Items[false]+31;',
  21272. 'end;',
  21273. 'begin',
  21274. ' b.Size[true]:=b.Size[false]+11;',
  21275. ' b.Speed[true]:=b.Speed[false]+12;',
  21276. ' b.Items[true]:=b.Items[false]+13;',
  21277. ' with b do begin',
  21278. ' Size[true]:=Size[false]+21;',
  21279. ' Speed[true]:=Speed[false]+22;',
  21280. ' Items[true]:=Items[false]+23;',
  21281. ' end;',
  21282. '']);
  21283. ConvertProgram;
  21284. CheckSource('TestClassHelper_Property_Array',
  21285. LinesToStr([ // statements
  21286. 'rtl.createClass(this, "TObject", null, function () {',
  21287. ' this.$init = function () {',
  21288. ' };',
  21289. ' this.$final = function () {',
  21290. ' };',
  21291. ' this.GetSpeed = function (Index) {',
  21292. ' var Result = 0;',
  21293. ' Result = $mod.TObjHelper.GetSize.call(this, false);',
  21294. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 11);',
  21295. ' this.SetSpeed(true, this.GetSpeed(false) + 12);',
  21296. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 21);',
  21297. ' this.SetSpeed(true, this.GetSpeed(false) + 22);',
  21298. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 31);',
  21299. ' this.SetSpeed(true, this.GetSpeed(false) + 32);',
  21300. ' return Result;',
  21301. ' };',
  21302. ' this.SetSpeed = function (Index, Value) {',
  21303. ' };',
  21304. '});',
  21305. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21306. ' this.GetSize = function (Index) {',
  21307. ' var Result = 0;',
  21308. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 11);',
  21309. ' this.SetSpeed(true, this.GetSpeed(false) + 12);',
  21310. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 21);',
  21311. ' this.SetSpeed(true, this.GetSpeed(false) + 22);',
  21312. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 31);',
  21313. ' this.SetSpeed(true, this.GetSpeed(false) + 32);',
  21314. ' return Result;',
  21315. ' };',
  21316. ' this.SetSize = function (Index, Value) {',
  21317. ' };',
  21318. '});',
  21319. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21320. ' this.DoIt = function () {',
  21321. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 11);',
  21322. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 21);',
  21323. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 31);',
  21324. ' };',
  21325. '});',
  21326. 'this.b = null;',
  21327. '']),
  21328. LinesToStr([ // $mod.$main
  21329. '$mod.TObjHelper.SetSize.call($mod.b, true, $mod.TObjHelper.GetSize.call($mod.b, false) + 11);',
  21330. '$mod.b.SetSpeed(true, $mod.b.GetSpeed(false) + 12);',
  21331. '$mod.TObjHelper.SetSize.call($mod.b, true, $mod.TObjHelper.GetSize.call($mod.b, false) + 13);',
  21332. 'var $with = $mod.b;',
  21333. '$mod.TObjHelper.SetSize.call($with, true, $mod.TObjHelper.GetSize.call($with, false) + 21);',
  21334. '$with.SetSpeed(true, $with.GetSpeed(false) + 22);',
  21335. '$mod.TObjHelper.SetSize.call($with, true, $mod.TObjHelper.GetSize.call($with, false) + 23);',
  21336. '']));
  21337. end;
  21338. procedure TTestModule.TestClassHelper_Property_Array_Default;
  21339. begin
  21340. StartProgram(false);
  21341. Add([
  21342. 'type',
  21343. ' TObject = class',
  21344. ' function GetSpeed(Index: boolean): word;',
  21345. ' procedure SetSpeed(Index: boolean; Value: word);',
  21346. ' end;',
  21347. ' TObjHelper = class helper for TObject',
  21348. ' property Speed[Index: boolean]: word read GetSpeed write SetSpeed; default;',
  21349. ' end;',
  21350. ' TBird = class',
  21351. ' end;',
  21352. ' TBirdHelper = class helper for TBird',
  21353. ' function GetSize(Index: word): boolean;',
  21354. ' procedure SetSize(Index: word; Value: boolean);',
  21355. ' property Size[Index: word]: boolean read GetSize write SetSize; default;',
  21356. ' end;',
  21357. 'function Tobject.GetSpeed(Index: boolean): word;',
  21358. 'begin',
  21359. ' Self[true]:=Self[false]+1;',
  21360. 'end;',
  21361. 'procedure Tobject.SetSpeed(Index: boolean; Value: word);',
  21362. 'begin',
  21363. 'end;',
  21364. 'function TBirdHelper.GetSize(Index: word): boolean;',
  21365. 'begin',
  21366. ' Self[1]:=not Self[2];',
  21367. 'end;',
  21368. 'procedure TBirdHelper.SetSize(Index: word; Value: boolean);',
  21369. 'begin',
  21370. 'end;',
  21371. 'var',
  21372. ' o: TObject;',
  21373. ' b: TBird;',
  21374. 'begin',
  21375. ' o[true]:=o[false]+1;',
  21376. ' b[3]:=not b[4];',
  21377. '']);
  21378. ConvertProgram;
  21379. CheckSource('TestClassHelper_Property_Array_Default',
  21380. LinesToStr([ // statements
  21381. 'rtl.createClass(this, "TObject", null, function () {',
  21382. ' this.$init = function () {',
  21383. ' };',
  21384. ' this.$final = function () {',
  21385. ' };',
  21386. ' this.GetSpeed = function (Index) {',
  21387. ' var Result = 0;',
  21388. ' this.SetSpeed(true, this.GetSpeed(false) + 1);',
  21389. ' return Result;',
  21390. ' };',
  21391. ' this.SetSpeed = function (Index, Value) {',
  21392. ' };',
  21393. '});',
  21394. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21395. '});',
  21396. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21397. '});',
  21398. 'rtl.createHelper(this, "TBirdHelper", null, function () {',
  21399. ' this.GetSize = function (Index) {',
  21400. ' var Result = false;',
  21401. ' $mod.TBirdHelper.SetSize.call(this, 1, !$mod.TBirdHelper.GetSize.call(this, 2));',
  21402. ' return Result;',
  21403. ' };',
  21404. ' this.SetSize = function (Index, Value) {',
  21405. ' };',
  21406. '});',
  21407. 'this.o = null;',
  21408. 'this.b = null;',
  21409. '']),
  21410. LinesToStr([ // $mod.$main
  21411. '$mod.o.SetSpeed(true, $mod.o.GetSpeed(false) + 1);',
  21412. '$mod.TBirdHelper.SetSize.call($mod.b, 3, !$mod.TBirdHelper.GetSize.call($mod.b, 4));',
  21413. '']));
  21414. end;
  21415. procedure TTestModule.TestClassHelper_Property_Array_DefaultDefault;
  21416. begin
  21417. StartProgram(false);
  21418. Add([
  21419. 'type',
  21420. ' TObject = class',
  21421. ' end;',
  21422. ' TObjHelper = class helper for TObject',
  21423. ' function GetItems(Index: word): TObject;',
  21424. ' procedure SetItems(Index: word; Value: TObject);',
  21425. ' property Items[Index: word]: TObject read GetItems write SetItems; default;',
  21426. ' end;',
  21427. 'function Tobjhelper.GetItems(Index: word): TObject;',
  21428. 'begin',
  21429. ' Self[1][2]:=Self[3][4];',
  21430. 'end;',
  21431. 'procedure Tobjhelper.SetItems(Index: word; Value: TObject);',
  21432. 'begin',
  21433. 'end;',
  21434. 'var',
  21435. ' o: TObject;',
  21436. 'begin',
  21437. ' o[1][2]:=o[3][4];',
  21438. '']);
  21439. ConvertProgram;
  21440. CheckSource('TestClassHelper_Property_Array_DefaultDefault',
  21441. LinesToStr([ // statements
  21442. 'rtl.createClass(this, "TObject", null, function () {',
  21443. ' this.$init = function () {',
  21444. ' };',
  21445. ' this.$final = function () {',
  21446. ' };',
  21447. '});',
  21448. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21449. ' this.GetItems = function (Index) {',
  21450. ' var Result = null;',
  21451. ' $mod.TObjHelper.SetItems.call($mod.TObjHelper.GetItems.call(this, 1), 2, $mod.TObjHelper.GetItems.call($mod.TObjHelper.GetItems.call(this, 3), 4));',
  21452. ' return Result;',
  21453. ' };',
  21454. ' this.SetItems = function (Index, Value) {',
  21455. ' };',
  21456. '});',
  21457. 'this.o = null;',
  21458. '']),
  21459. LinesToStr([ // $mod.$main
  21460. '$mod.TObjHelper.SetItems.call($mod.TObjHelper.GetItems.call($mod.o, 1), 2, $mod.TObjHelper.GetItems.call($mod.TObjHelper.GetItems.call($mod.o, 3), 4));',
  21461. '']));
  21462. end;
  21463. procedure TTestModule.TestClassHelper_ClassProperty;
  21464. begin
  21465. StartProgram(false);
  21466. Add([
  21467. 'type',
  21468. ' TObject = class',
  21469. ' class var FSize: word;',
  21470. ' class function GetSpeed: word;',
  21471. ' class procedure SetSpeed(Value: word); virtual; abstract;',
  21472. ' end;',
  21473. ' TObjHelper = class helper for TObject',
  21474. ' class function GetLeft: word;',
  21475. ' class procedure SetLeft(Value: word);',
  21476. ' class property Size: word read FSize write FSize;',
  21477. ' class property Speed: word read GetSpeed write SetSpeed;',
  21478. ' class property Left: word read GetLeft write SetLeft;',
  21479. ' end;',
  21480. ' TBird = class',
  21481. ' class property NotRight: word read GetLeft write SetLeft;',
  21482. ' class procedure DoIt;',
  21483. ' end;',
  21484. ' TBirdClass = class of TBird;',
  21485. 'class function Tobject.GetSpeed: word;',
  21486. 'begin',
  21487. ' Size:=Size+11;',
  21488. ' Speed:=Speed+12;',
  21489. ' Left:=Left+13;',
  21490. ' Self.Size:=Self.Size+21;',
  21491. ' Self.Speed:=Self.Speed+22;',
  21492. ' Self.Left:=Self.Left+23;',
  21493. ' with Self do begin',
  21494. ' Size:=Size+31;',
  21495. ' Speed:=Speed+32;',
  21496. ' Left:=Left+33;',
  21497. ' end;',
  21498. 'end;',
  21499. 'class function TObjHelper.GetLeft: word;',
  21500. 'begin',
  21501. ' Size:=Size+11;',
  21502. ' Speed:=Speed+12;',
  21503. ' Left:=Left+13;',
  21504. ' Self.Size:=Self.Size+21;',
  21505. ' Self.Speed:=Self.Speed+22;',
  21506. ' Self.Left:=Self.Left+23;',
  21507. ' with Self do begin',
  21508. ' Size:=Size+31;',
  21509. ' Speed:=Speed+32;',
  21510. ' Left:=Left+33;',
  21511. ' end;',
  21512. 'end;',
  21513. 'class procedure TObjHelper.SetLeft(Value: word);',
  21514. 'begin',
  21515. 'end;',
  21516. 'class procedure TBird.DoIt;',
  21517. 'begin',
  21518. ' NotRight:=NotRight+11;',
  21519. ' Self.NotRight:=Self.NotRight+21;',
  21520. ' with Self do NotRight:=NotRight+31;',
  21521. 'end;',
  21522. 'var',
  21523. ' b: TBird;',
  21524. ' c: TBirdClass;',
  21525. 'begin',
  21526. ' b.Size:=b.Size+11;',
  21527. ' b.Speed:=b.Speed+12;',
  21528. ' b.Left:=b.Left+13;',
  21529. ' b.NotRight:=b.NotRight+14;',
  21530. ' with b do begin',
  21531. ' Size:=Size+31;',
  21532. ' Speed:=Speed+32;',
  21533. ' Left:=Left+33;',
  21534. ' NotRight:=NotRight+34;',
  21535. ' end;',
  21536. ' c.Size:=c.Size+11;',
  21537. ' c.Speed:=c.Speed+12;',
  21538. ' c.Left:=c.Left+13;',
  21539. ' c.NotRight:=c.NotRight+14;',
  21540. ' with c do begin',
  21541. ' Size:=Size+31;',
  21542. ' Speed:=Speed+32;',
  21543. ' Left:=Left+33;',
  21544. ' NotRight:=NotRight+34;',
  21545. ' end;',
  21546. ' tbird.Size:=tbird.Size+11;',
  21547. ' tbird.Speed:=tbird.Speed+12;',
  21548. ' tbird.Left:=tbird.Left+13;',
  21549. ' tbird.NotRight:=tbird.NotRight+14;',
  21550. ' with tbird do begin',
  21551. ' Size:=Size+31;',
  21552. ' Speed:=Speed+32;',
  21553. ' Left:=Left+33;',
  21554. ' NotRight:=NotRight+34;',
  21555. ' end;',
  21556. '']);
  21557. ConvertProgram;
  21558. CheckSource('TestClassHelper_ClassProperty',
  21559. LinesToStr([ // statements
  21560. 'rtl.createClass(this, "TObject", null, function () {',
  21561. ' this.FSize = 0;',
  21562. ' this.$init = function () {',
  21563. ' };',
  21564. ' this.$final = function () {',
  21565. ' };',
  21566. ' this.GetSpeed = function () {',
  21567. ' var Result = 0;',
  21568. ' $mod.TObject.FSize = this.FSize + 11;',
  21569. ' this.SetSpeed(this.GetSpeed() + 12);',
  21570. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 13);',
  21571. ' $mod.TObject.FSize = this.FSize + 21;',
  21572. ' this.SetSpeed(this.GetSpeed() + 22);',
  21573. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 23);',
  21574. ' $mod.TObject.FSize = this.FSize + 31;',
  21575. ' this.SetSpeed(this.GetSpeed() + 32);',
  21576. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 33);',
  21577. ' return Result;',
  21578. ' };',
  21579. '});',
  21580. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21581. ' this.GetLeft = function () {',
  21582. ' var Result = 0;',
  21583. ' $mod.TObject.FSize = this.FSize + 11;',
  21584. ' this.SetSpeed(this.GetSpeed() + 12);',
  21585. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 13);',
  21586. ' $mod.TObject.FSize = this.FSize + 21;',
  21587. ' this.SetSpeed(this.GetSpeed() + 22);',
  21588. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 23);',
  21589. ' $mod.TObject.FSize = this.FSize + 31;',
  21590. ' this.SetSpeed(this.GetSpeed() + 32);',
  21591. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 33);',
  21592. ' return Result;',
  21593. ' };',
  21594. ' this.SetLeft = function (Value) {',
  21595. ' };',
  21596. '});',
  21597. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21598. ' this.DoIt = function () {',
  21599. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 11);',
  21600. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 21);',
  21601. ' $mod.TObjHelper.SetLeft.call(this, $mod.TObjHelper.GetLeft.call(this) + 31);',
  21602. ' };',
  21603. '});',
  21604. 'this.b = null;',
  21605. 'this.c = null;',
  21606. '']),
  21607. LinesToStr([ // $mod.$main
  21608. '$mod.TObject.FSize = $mod.b.FSize + 11;',
  21609. '$mod.b.$class.SetSpeed($mod.b.$class.GetSpeed() + 12);',
  21610. '$mod.TObjHelper.SetLeft.call($mod.b.$class, $mod.TObjHelper.GetLeft.call($mod.b.$class) + 13);',
  21611. '$mod.TObjHelper.SetLeft.call($mod.b.$class, $mod.TObjHelper.GetLeft.call($mod.b.$class) + 14);',
  21612. 'var $with = $mod.b;',
  21613. '$mod.TObject.FSize = $with.FSize + 31;',
  21614. '$with.$class.SetSpeed($with.$class.GetSpeed() + 32);',
  21615. '$mod.TObjHelper.SetLeft.call($with.$class, $mod.TObjHelper.GetLeft.call($with.$class) + 33);',
  21616. '$mod.TObjHelper.SetLeft.call($with.$class, $mod.TObjHelper.GetLeft.call($with.$class) + 34);',
  21617. '$mod.TObject.FSize = $mod.c.FSize + 11;',
  21618. '$mod.c.SetSpeed($mod.c.GetSpeed() + 12);',
  21619. '$mod.TObjHelper.SetLeft.call($mod.c, $mod.TObjHelper.GetLeft.call($mod.c) + 13);',
  21620. '$mod.TObjHelper.SetLeft.call($mod.c, $mod.TObjHelper.GetLeft.call($mod.c) + 14);',
  21621. 'var $with1 = $mod.c;',
  21622. '$mod.TObject.FSize = $with1.FSize + 31;',
  21623. '$with1.SetSpeed($with1.GetSpeed() + 32);',
  21624. '$mod.TObjHelper.SetLeft.call($with1, $mod.TObjHelper.GetLeft.call($with1) + 33);',
  21625. '$mod.TObjHelper.SetLeft.call($with1, $mod.TObjHelper.GetLeft.call($with1) + 34);',
  21626. '$mod.TObject.FSize = $mod.TBird.FSize + 11;',
  21627. '$mod.TBird.SetSpeed($mod.TBird.GetSpeed() + 12);',
  21628. '$mod.TObjHelper.SetLeft.call($mod.TBird, $mod.TObjHelper.GetLeft.call($mod.TBird) + 13);',
  21629. '$mod.TObjHelper.SetLeft.call($mod.TBird, $mod.TObjHelper.GetLeft.call($mod.TBird) + 14);',
  21630. 'var $with2 = $mod.TBird;',
  21631. '$mod.TObject.FSize = $with2.FSize + 31;',
  21632. '$with2.SetSpeed($with2.GetSpeed() + 32);',
  21633. '$mod.TObjHelper.SetLeft.call($mod.TBird, $mod.TObjHelper.GetLeft.call($mod.TBird) + 33);',
  21634. '$mod.TObjHelper.SetLeft.call($mod.TBird, $mod.TObjHelper.GetLeft.call($mod.TBird) + 34);',
  21635. '']));
  21636. end;
  21637. procedure TTestModule.TestClassHelper_ClassPropertyStatic;
  21638. begin
  21639. StartProgram(false);
  21640. Add([
  21641. 'type',
  21642. ' TObject = class',
  21643. ' class function GetSpeed: word; static;',
  21644. ' class procedure SetSpeed(Value: word); static;',
  21645. ' end;',
  21646. ' TObjHelper = class helper for TObject',
  21647. ' class function GetLeft: word; static;',
  21648. ' class procedure SetLeft(Value: word); static;',
  21649. ' class property Speed: word read GetSpeed write SetSpeed;',
  21650. ' class property Left: word read GetLeft write SetLeft;',
  21651. ' end;',
  21652. ' TBird = class',
  21653. ' class property NotRight: word read GetLeft write SetLeft;',
  21654. ' class procedure DoIt; static;',
  21655. ' class procedure DoSome;',
  21656. ' end;',
  21657. ' TBirdClass = class of TBird;',
  21658. 'class function Tobject.GetSpeed: word;',
  21659. 'begin',
  21660. ' Speed:=Speed+12;',
  21661. ' Left:=Left+13;',
  21662. 'end;',
  21663. 'class procedure TObject.SetSpeed(Value: word);',
  21664. 'begin',
  21665. 'end;',
  21666. 'class function TObjHelper.GetLeft: word;',
  21667. 'begin',
  21668. ' Speed:=Speed+12;',
  21669. ' Left:=Left+13;',
  21670. 'end;',
  21671. 'class procedure TObjHelper.SetLeft(Value: word);',
  21672. 'begin',
  21673. 'end;',
  21674. 'class procedure TBird.DoIt;',
  21675. 'begin',
  21676. ' NotRight:=NotRight+11;',
  21677. 'end;',
  21678. 'class procedure TBird.DoSome;',
  21679. 'begin',
  21680. ' Speed:=Speed+12;',
  21681. ' Left:=Left+13;',
  21682. ' Self.Speed:=Self.Speed+22;',
  21683. ' Self.Left:=Self.Left+23;',
  21684. ' with Self do begin',
  21685. ' Speed:=Speed+32;',
  21686. ' Left:=Left+33;',
  21687. ' end;',
  21688. ' NotRight:=NotRight+11;',
  21689. ' Self.NotRight:=Self.NotRight+21;',
  21690. ' with Self do NotRight:=NotRight+31;',
  21691. 'end;',
  21692. 'var',
  21693. ' b: TBird;',
  21694. ' c: TBirdClass;',
  21695. 'begin',
  21696. ' b.Speed:=b.Speed+12;',
  21697. ' b.Left:=b.Left+13;',
  21698. ' b.NotRight:=b.NotRight+14;',
  21699. ' with b do begin',
  21700. ' Speed:=Speed+32;',
  21701. ' Left:=Left+33;',
  21702. ' NotRight:=NotRight+34;',
  21703. ' end;',
  21704. ' c.Speed:=c.Speed+12;',
  21705. ' c.Left:=c.Left+13;',
  21706. ' c.NotRight:=c.NotRight+14;',
  21707. ' with c do begin',
  21708. ' Speed:=Speed+32;',
  21709. ' Left:=Left+33;',
  21710. ' NotRight:=NotRight+34;',
  21711. ' end;',
  21712. ' tbird.Speed:=tbird.Speed+12;',
  21713. ' tbird.Left:=tbird.Left+13;',
  21714. ' tbird.NotRight:=tbird.NotRight+14;',
  21715. ' with tbird do begin',
  21716. ' Speed:=Speed+32;',
  21717. ' Left:=Left+33;',
  21718. ' NotRight:=NotRight+34;',
  21719. ' end;',
  21720. '']);
  21721. ConvertProgram;
  21722. CheckSource('TestClassHelper_ClassPropertyStatic',
  21723. LinesToStr([ // statements
  21724. 'rtl.createClass(this, "TObject", null, function () {',
  21725. ' this.$init = function () {',
  21726. ' };',
  21727. ' this.$final = function () {',
  21728. ' };',
  21729. ' this.GetSpeed = function () {',
  21730. ' var Result = 0;',
  21731. ' $mod.TObject.SetSpeed($mod.TObject.GetSpeed() + 12);',
  21732. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 13);',
  21733. ' return Result;',
  21734. ' };',
  21735. ' this.SetSpeed = function (Value) {',
  21736. ' };',
  21737. '});',
  21738. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21739. ' this.GetLeft = function () {',
  21740. ' var Result = 0;',
  21741. ' $mod.TObject.SetSpeed($mod.TObject.GetSpeed() + 12);',
  21742. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 13);',
  21743. ' return Result;',
  21744. ' };',
  21745. ' this.SetLeft = function (Value) {',
  21746. ' };',
  21747. '});',
  21748. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21749. ' this.DoIt = function () {',
  21750. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 11);',
  21751. ' };',
  21752. ' this.DoSome = function () {',
  21753. ' this.SetSpeed(this.GetSpeed() + 12);',
  21754. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 13);',
  21755. ' this.SetSpeed(this.GetSpeed() + 22);',
  21756. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 23);',
  21757. ' this.SetSpeed(this.GetSpeed() + 32);',
  21758. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 33);',
  21759. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 11);',
  21760. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 21);',
  21761. ' $mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 31);',
  21762. ' };',
  21763. '});',
  21764. 'this.b = null;',
  21765. 'this.c = null;',
  21766. '']),
  21767. LinesToStr([ // $mod.$main
  21768. '$mod.b.SetSpeed($mod.b.GetSpeed() + 12);',
  21769. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 13);',
  21770. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 14);',
  21771. 'var $with = $mod.b;',
  21772. '$with.SetSpeed($with.GetSpeed() + 32);',
  21773. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 33);',
  21774. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 34);',
  21775. '$mod.c.SetSpeed($mod.c.GetSpeed() + 12);',
  21776. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 13);',
  21777. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 14);',
  21778. 'var $with1 = $mod.c;',
  21779. '$with1.SetSpeed($with1.GetSpeed() + 32);',
  21780. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 33);',
  21781. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 34);',
  21782. '$mod.TBird.SetSpeed($mod.TBird.GetSpeed() + 12);',
  21783. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 13);',
  21784. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 14);',
  21785. 'var $with2 = $mod.TBird;',
  21786. '$with2.SetSpeed($with2.GetSpeed() + 32);',
  21787. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 33);',
  21788. '$mod.TObjHelper.SetLeft($mod.TObjHelper.GetLeft() + 34);',
  21789. '']));
  21790. end;
  21791. procedure TTestModule.TestClassHelper_ClassProperty_Array;
  21792. begin
  21793. StartProgram(false);
  21794. Add([
  21795. 'type',
  21796. ' TObject = class',
  21797. ' class function GetSpeed(Index: boolean): word;',
  21798. ' class procedure SetSpeed(Index: boolean; Value: word); virtual; abstract;',
  21799. ' end;',
  21800. ' TObjHelper = class helper for TObject',
  21801. ' class function GetSize(Index: boolean): word;',
  21802. ' class procedure SetSize(Index: boolean; Value: word);',
  21803. ' class property Size[Index: boolean]: word read GetSize write SetSize;',
  21804. ' class property Speed[Index: boolean]: word read GetSpeed write SetSpeed;',
  21805. ' end;',
  21806. ' TBird = class',
  21807. ' class property Items[Index: boolean]: word read GetSize write SetSize;',
  21808. ' class procedure DoIt;',
  21809. ' end;',
  21810. ' TBirdClass = class of TBird;',
  21811. 'class function Tobject.GetSpeed(Index: boolean): word;',
  21812. 'begin',
  21813. ' Size[true]:=Size[false]+11;',
  21814. ' Speed[true]:=Speed[false]+12;',
  21815. ' Self.Size[true]:=Self.Size[false]+21;',
  21816. ' Self.Speed[true]:=Self.Speed[false]+22;',
  21817. ' with Self do begin',
  21818. ' Size[true]:=Size[false]+31;',
  21819. ' Speed[true]:=Speed[false]+32;',
  21820. ' end;',
  21821. 'end;',
  21822. 'class function TObjHelper.GetSize(Index: boolean): word;',
  21823. 'begin',
  21824. ' Size[true]:=Size[false]+11;',
  21825. ' Speed[true]:=Speed[false]+12;',
  21826. ' Self.Size[true]:=Self.Size[false]+21;',
  21827. ' Self.Speed[true]:=Self.Speed[false]+22;',
  21828. ' with Self do begin',
  21829. ' Size[true]:=Size[false]+31;',
  21830. ' Speed[true]:=Speed[false]+32;',
  21831. ' end;',
  21832. 'end;',
  21833. 'class procedure TObjHelper.SetSize(Index: boolean; Value: word);',
  21834. 'begin',
  21835. 'end;',
  21836. 'class procedure TBird.DoIt;',
  21837. 'begin',
  21838. ' Items[true]:=Items[false]+11;',
  21839. ' Self.Items[true]:=Self.Items[false]+21;',
  21840. ' with Self do Items[true]:=Items[false]+31;',
  21841. 'end;',
  21842. 'var',
  21843. ' b: TBird;',
  21844. ' c: TBirdClass;',
  21845. 'begin',
  21846. ' b.Size[true]:=b.Size[false]+11;',
  21847. ' b.Speed[true]:=b.Speed[false]+12;',
  21848. ' b.Items[true]:=b.Items[false]+13;',
  21849. ' with b do begin',
  21850. ' Size[true]:=Size[false]+21;',
  21851. ' Speed[true]:=Speed[false]+22;',
  21852. ' Items[true]:=Items[false]+23;',
  21853. ' end;',
  21854. ' c.Size[true]:=c.Size[false]+11;',
  21855. ' c.Speed[true]:=c.Speed[false]+12;',
  21856. ' c.Items[true]:=c.Items[false]+13;',
  21857. ' with c do begin',
  21858. ' Size[true]:=Size[false]+21;',
  21859. ' Speed[true]:=Speed[false]+22;',
  21860. ' Items[true]:=Items[false]+23;',
  21861. ' end;',
  21862. ' TBird.Size[true]:=TBird.Size[false]+11;',
  21863. ' TBird.Speed[true]:=TBird.Speed[false]+12;',
  21864. ' TBird.Items[true]:=TBird.Items[false]+13;',
  21865. ' with TBird do begin',
  21866. ' Size[true]:=Size[false]+21;',
  21867. ' Speed[true]:=Speed[false]+22;',
  21868. ' Items[true]:=Items[false]+23;',
  21869. ' end;',
  21870. '']);
  21871. ConvertProgram;
  21872. CheckSource('TestClassHelper_ClassProperty_Array',
  21873. LinesToStr([ // statements
  21874. 'rtl.createClass(this, "TObject", null, function () {',
  21875. ' this.$init = function () {',
  21876. ' };',
  21877. ' this.$final = function () {',
  21878. ' };',
  21879. ' this.GetSpeed = function (Index) {',
  21880. ' var Result = 0;',
  21881. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 11);',
  21882. ' this.SetSpeed(true, this.GetSpeed(false) + 12);',
  21883. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 21);',
  21884. ' this.SetSpeed(true, this.GetSpeed(false) + 22);',
  21885. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 31);',
  21886. ' this.SetSpeed(true, this.GetSpeed(false) + 32);',
  21887. ' return Result;',
  21888. ' };',
  21889. '});',
  21890. 'rtl.createHelper(this, "TObjHelper", null, function () {',
  21891. ' this.GetSize = function (Index) {',
  21892. ' var Result = 0;',
  21893. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 11);',
  21894. ' this.SetSpeed(true, this.GetSpeed(false) + 12);',
  21895. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 21);',
  21896. ' this.SetSpeed(true, this.GetSpeed(false) + 22);',
  21897. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 31);',
  21898. ' this.SetSpeed(true, this.GetSpeed(false) + 32);',
  21899. ' return Result;',
  21900. ' };',
  21901. ' this.SetSize = function (Index, Value) {',
  21902. ' };',
  21903. '});',
  21904. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21905. ' this.DoIt = function () {',
  21906. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 11);',
  21907. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 21);',
  21908. ' $mod.TObjHelper.SetSize.call(this, true, $mod.TObjHelper.GetSize.call(this, false) + 31);',
  21909. ' };',
  21910. '});',
  21911. 'this.b = null;',
  21912. 'this.c = null;',
  21913. '']),
  21914. LinesToStr([ // $mod.$main
  21915. '$mod.TObjHelper.SetSize.call($mod.b.$class, true, $mod.TObjHelper.GetSize.call($mod.b.$class, false) + 11);',
  21916. '$mod.b.$class.SetSpeed(true, $mod.b.$class.GetSpeed(false) + 12);',
  21917. '$mod.TObjHelper.SetSize.call($mod.b.$class, true, $mod.TObjHelper.GetSize.call($mod.b.$class, false) + 13);',
  21918. 'var $with = $mod.b;',
  21919. '$mod.TObjHelper.SetSize.call($with.$class, true, $mod.TObjHelper.GetSize.call($with.$class, false) + 21);',
  21920. '$with.$class.SetSpeed(true, $with.$class.GetSpeed(false) + 22);',
  21921. '$mod.TObjHelper.SetSize.call($with.$class, true, $mod.TObjHelper.GetSize.call($with.$class, false) + 23);',
  21922. '$mod.TObjHelper.SetSize.call($mod.c, true, $mod.TObjHelper.GetSize.call($mod.c, false) + 11);',
  21923. '$mod.c.SetSpeed(true, $mod.c.GetSpeed(false) + 12);',
  21924. '$mod.TObjHelper.SetSize.call($mod.c, true, $mod.TObjHelper.GetSize.call($mod.c, false) + 13);',
  21925. 'var $with1 = $mod.c;',
  21926. '$mod.TObjHelper.SetSize.call($with1, true, $mod.TObjHelper.GetSize.call($with1, false) + 21);',
  21927. '$with1.SetSpeed(true, $with1.GetSpeed(false) + 22);',
  21928. '$mod.TObjHelper.SetSize.call($with1, true, $mod.TObjHelper.GetSize.call($with1, false) + 23);',
  21929. '$mod.TObjHelper.SetSize.call($mod.TBird, true, $mod.TObjHelper.GetSize.call($mod.TBird, false) + 11);',
  21930. '$mod.TBird.SetSpeed(true, $mod.TBird.GetSpeed(false) + 12);',
  21931. '$mod.TObjHelper.SetSize.call($mod.TBird, true, $mod.TObjHelper.GetSize.call($mod.TBird, false) + 13);',
  21932. 'var $with2 = $mod.TBird;',
  21933. '$mod.TObjHelper.SetSize.call($mod.TBird, true, $mod.TObjHelper.GetSize.call($mod.TBird, false) + 21);',
  21934. '$with2.SetSpeed(true, $with2.GetSpeed(false) + 22);',
  21935. '$mod.TObjHelper.SetSize.call($mod.TBird, true, $mod.TObjHelper.GetSize.call($mod.TBird, false) + 23);',
  21936. '']));
  21937. end;
  21938. procedure TTestModule.TestClassHelper_ForIn;
  21939. begin
  21940. StartProgram(false);
  21941. Add([
  21942. 'type',
  21943. ' TObject = class end;',
  21944. ' TItem = TObject;',
  21945. ' TEnumerator = class',
  21946. ' FCurrent: TItem;',
  21947. ' property Current: TItem read FCurrent;',
  21948. ' function MoveNext: boolean;',
  21949. ' end;',
  21950. ' TBird = class',
  21951. ' end;',
  21952. ' TBirdHelper = class helper for TBird',
  21953. ' function GetEnumerator: TEnumerator;',
  21954. ' end;',
  21955. 'function TEnumerator.MoveNext: boolean;',
  21956. 'begin',
  21957. 'end;',
  21958. 'function TBirdHelper.GetEnumerator: TEnumerator;',
  21959. 'begin',
  21960. 'end;',
  21961. 'var',
  21962. ' b: TBird;',
  21963. ' i, i2: TItem;',
  21964. 'begin',
  21965. ' for i in b do i2:=i;']);
  21966. ConvertProgram;
  21967. CheckSource('TestClassHelper_ForIn',
  21968. LinesToStr([ // statements
  21969. 'rtl.createClass(this, "TObject", null, function () {',
  21970. ' this.$init = function () {',
  21971. ' };',
  21972. ' this.$final = function () {',
  21973. ' };',
  21974. '});',
  21975. 'rtl.createClass(this, "TEnumerator", this.TObject, function () {',
  21976. ' this.$init = function () {',
  21977. ' $mod.TObject.$init.call(this);',
  21978. ' this.FCurrent = null;',
  21979. ' };',
  21980. ' this.$final = function () {',
  21981. ' this.FCurrent = undefined;',
  21982. ' $mod.TObject.$final.call(this);',
  21983. ' };',
  21984. ' this.MoveNext = function () {',
  21985. ' var Result = false;',
  21986. ' return Result;',
  21987. ' };',
  21988. '});',
  21989. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  21990. '});',
  21991. 'rtl.createHelper(this, "TBirdHelper", null, function () {',
  21992. ' this.GetEnumerator = function () {',
  21993. ' var Result = null;',
  21994. ' return Result;',
  21995. ' };',
  21996. '});',
  21997. 'this.b = null;',
  21998. 'this.i = null;',
  21999. 'this.i2 = null;'
  22000. ]),
  22001. LinesToStr([ // $mod.$main
  22002. 'var $in = $mod.TBirdHelper.GetEnumerator.call($mod.b);',
  22003. 'try {',
  22004. ' while ($in.MoveNext()){',
  22005. ' $mod.i = $in.FCurrent;',
  22006. ' $mod.i2 = $mod.i;',
  22007. ' }',
  22008. '} finally {',
  22009. ' $in = rtl.freeLoc($in)',
  22010. '};',
  22011. '']));
  22012. end;
  22013. procedure TTestModule.TestClassHelper_PassProperty;
  22014. begin
  22015. StartProgram(false);
  22016. Add([
  22017. 'type',
  22018. ' TObject = class',
  22019. ' FField: TObject;',
  22020. ' property Field: TObject read FField write FField;',
  22021. ' end;',
  22022. ' THelper = class helper for TObject',
  22023. ' procedure Fly;',
  22024. ' class procedure Run;',
  22025. ' class procedure Jump; static;',
  22026. ' end;',
  22027. 'procedure THelper.Fly;',
  22028. 'begin',
  22029. ' Field.Fly;',
  22030. ' Field.Run;',
  22031. ' Field.Jump;',
  22032. ' with Field do begin',
  22033. ' Fly;',
  22034. ' Run;',
  22035. ' Jump;',
  22036. ' end;',
  22037. 'end;',
  22038. 'class procedure THelper.Run;',
  22039. 'begin',
  22040. 'end;',
  22041. 'class procedure THelper.Jump;',
  22042. 'begin',
  22043. 'end;',
  22044. 'var',
  22045. ' b: TObject;',
  22046. 'begin',
  22047. ' b.Field.Fly;',
  22048. ' b.Field.Run;',
  22049. ' b.Field.Jump;',
  22050. ' with b do begin',
  22051. ' Field.Run;',
  22052. ' Field.Fly;',
  22053. ' Field.Jump;',
  22054. ' end;',
  22055. ' with b.Field do begin',
  22056. ' Run;',
  22057. ' Fly;',
  22058. ' Jump;',
  22059. ' end;',
  22060. '']);
  22061. ConvertProgram;
  22062. CheckSource('TestClassHelper_PassProperty',
  22063. LinesToStr([ // statements
  22064. 'rtl.createClass(this, "TObject", null, function () {',
  22065. ' this.$init = function () {',
  22066. ' this.FField = null;',
  22067. ' };',
  22068. ' this.$final = function () {',
  22069. ' this.FField = undefined;',
  22070. ' };',
  22071. '});',
  22072. 'rtl.createHelper(this, "THelper", null, function () {',
  22073. ' this.Fly = function () {',
  22074. ' $mod.THelper.Fly.call(this.FField);',
  22075. ' $mod.THelper.Run.call(this.FField.$class);',
  22076. ' $mod.THelper.Jump();',
  22077. ' var $with = this.FField;',
  22078. ' $mod.THelper.Fly.call($with);',
  22079. ' $mod.THelper.Run.call($with.$class);',
  22080. ' $mod.THelper.Jump();',
  22081. ' };',
  22082. ' this.Run = function () {',
  22083. ' };',
  22084. ' this.Jump = function () {',
  22085. ' };',
  22086. '});',
  22087. 'this.b = null;',
  22088. '']),
  22089. LinesToStr([ // $mod.$main
  22090. '$mod.THelper.Fly.call($mod.b.FField);',
  22091. '$mod.THelper.Run.call($mod.b.FField.$class);',
  22092. '$mod.THelper.Jump();',
  22093. 'var $with = $mod.b;',
  22094. '$mod.THelper.Run.call($with.FField.$class);',
  22095. '$mod.THelper.Fly.call($with.FField);',
  22096. '$mod.THelper.Jump();',
  22097. 'var $with1 = $mod.b.FField;',
  22098. '$mod.THelper.Run.call($with1.$class);',
  22099. '$mod.THelper.Fly.call($with1);',
  22100. '$mod.THelper.Jump();',
  22101. '']));
  22102. end;
  22103. procedure TTestModule.TestExtClassHelper_ClassVar;
  22104. begin
  22105. StartProgram(false);
  22106. Add([
  22107. '{$modeswitch externalclass}',
  22108. 'type',
  22109. ' TExtA = class external name ''ExtObj''',
  22110. ' end;',
  22111. ' THelper = class helper for TExtA',
  22112. ' const',
  22113. ' One = 1;',
  22114. ' Two: word = 2;',
  22115. ' class var',
  22116. ' Glob: word;',
  22117. ' function Foo(w: word): word;',
  22118. ' class function Bar(w: word): word; static;',
  22119. ' end;',
  22120. 'function THelper.foo(w: word): word;',
  22121. 'begin',
  22122. ' Result:=w;',
  22123. ' Two:=One+w;',
  22124. ' Glob:=Glob;',
  22125. ' Result:=Self.Glob;',
  22126. ' Self.Glob:=Self.Glob;',
  22127. ' with Self do Glob:=Glob;',
  22128. 'end;',
  22129. 'class function THelper.bar(w: word): word;',
  22130. 'begin',
  22131. ' Result:=w;',
  22132. ' Two:=One;',
  22133. ' Glob:=Glob;',
  22134. 'end;',
  22135. 'var o: TExtA;',
  22136. 'begin',
  22137. ' texta.two:=texta.one;',
  22138. ' texta.Glob:=texta.Glob;',
  22139. ' with texta do begin',
  22140. ' two:=one;',
  22141. ' Glob:=Glob;',
  22142. ' end;',
  22143. ' o.two:=o.one;',
  22144. ' o.Glob:=o.Glob;',
  22145. ' with o do begin',
  22146. ' two:=one;',
  22147. ' Glob:=Glob;',
  22148. ' end;',
  22149. '']);
  22150. ConvertProgram;
  22151. CheckSource('TestExtClassHelper_ClassVar',
  22152. LinesToStr([ // statements
  22153. 'rtl.createHelper(this, "THelper", null, function () {',
  22154. ' this.One = 1;',
  22155. ' this.Two = 2;',
  22156. ' this.Glob = 0;',
  22157. ' this.Foo = function (w) {',
  22158. ' var Result = 0;',
  22159. ' Result = w;',
  22160. ' $mod.THelper.Two = 1 + w;',
  22161. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22162. ' Result = $mod.THelper.Glob;',
  22163. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22164. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22165. ' return Result;',
  22166. ' };',
  22167. ' this.Bar = function (w) {',
  22168. ' var Result = 0;',
  22169. ' Result = w;',
  22170. ' $mod.THelper.Two = 1;',
  22171. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22172. ' return Result;',
  22173. ' };',
  22174. '});',
  22175. 'this.o = null;',
  22176. '']),
  22177. LinesToStr([ // $mod.$main
  22178. '$mod.THelper.Two = 1;',
  22179. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22180. '$mod.THelper.Two = 1;',
  22181. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22182. '$mod.THelper.Two = 1;',
  22183. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22184. 'var $with = $mod.o;',
  22185. '$mod.THelper.Two = 1;',
  22186. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22187. '']));
  22188. end;
  22189. procedure TTestModule.TestExtClassHelper_Method_Call;
  22190. begin
  22191. StartProgram(false);
  22192. Add([
  22193. '{$modeswitch externalclass}',
  22194. 'type',
  22195. ' TFly = function(w: word): word of object;',
  22196. ' TExtA = class external name ''ExtObj''',
  22197. ' procedure Run(w: word = 10);',
  22198. ' end;',
  22199. ' THelper = class helper for TExtA',
  22200. ' function Foo(w: word = 1): word;',
  22201. ' function Fly(w: word = 2): word; external name ''Fly'';',
  22202. ' end;',
  22203. 'var p: TFly;',
  22204. 'function THelper.foo(w: word): word;',
  22205. 'begin',
  22206. ' Run;',
  22207. ' Run();',
  22208. ' Run(11);',
  22209. ' Foo;',
  22210. ' Foo();',
  22211. ' Foo(12);',
  22212. ' Self.Foo;',
  22213. ' Self.Foo();',
  22214. ' Self.Foo(13);',
  22215. ' Fly;',
  22216. ' Fly();',
  22217. ' with Self do begin',
  22218. ' Foo;',
  22219. ' Foo();',
  22220. ' Foo(14);',
  22221. ' Fly;',
  22222. ' Fly();',
  22223. ' end;',
  22224. ' p:=@Fly;',
  22225. 'end;',
  22226. 'var Obj: TExtA;',
  22227. 'begin',
  22228. ' obj.Foo;',
  22229. ' obj.Foo();',
  22230. ' obj.Foo(21);',
  22231. ' obj.Fly;',
  22232. ' obj.Fly();',
  22233. ' with obj do begin',
  22234. ' Foo;',
  22235. ' Foo();',
  22236. ' Foo(22);',
  22237. ' Fly;',
  22238. ' Fly();',
  22239. ' end;',
  22240. ' p:[email protected];',
  22241. '']);
  22242. ConvertProgram;
  22243. CheckSource('TestExtClassHelper_Method_Call',
  22244. LinesToStr([ // statements
  22245. 'rtl.createHelper(this, "THelper", null, function () {',
  22246. ' this.Foo = function (w) {',
  22247. ' var Result = 0;',
  22248. ' this.Run(10);',
  22249. ' this.Run(10);',
  22250. ' this.Run(11);',
  22251. ' $mod.THelper.Foo.call(this, 1);',
  22252. ' $mod.THelper.Foo.call(this, 1);',
  22253. ' $mod.THelper.Foo.call(this, 12);',
  22254. ' $mod.THelper.Foo.call(this, 1);',
  22255. ' $mod.THelper.Foo.call(this, 1);',
  22256. ' $mod.THelper.Foo.call(this, 13);',
  22257. ' this.Fly(2);',
  22258. ' this.Fly(2);',
  22259. ' $mod.THelper.Foo.call(this, 1);',
  22260. ' $mod.THelper.Foo.call(this, 1);',
  22261. ' $mod.THelper.Foo.call(this, 14);',
  22262. ' this.Fly(2);',
  22263. ' this.Fly(2);',
  22264. ' $mod.p = rtl.createCallback(this, "Fly");',
  22265. ' return Result;',
  22266. ' };',
  22267. '});',
  22268. 'this.p = null;',
  22269. 'this.Obj = null;',
  22270. '']),
  22271. LinesToStr([ // $mod.$main
  22272. '$mod.THelper.Foo.call($mod.Obj, 1);',
  22273. '$mod.THelper.Foo.call($mod.Obj, 1);',
  22274. '$mod.THelper.Foo.call($mod.Obj, 21);',
  22275. '$mod.Obj.Fly(2);',
  22276. '$mod.Obj.Fly(2);',
  22277. 'var $with = $mod.Obj;',
  22278. '$mod.THelper.Foo.call($with, 1);',
  22279. '$mod.THelper.Foo.call($with, 1);',
  22280. '$mod.THelper.Foo.call($with, 22);',
  22281. '$with.Fly(2);',
  22282. '$with.Fly(2);',
  22283. '$mod.p = rtl.createCallback($mod.Obj, "Fly");',
  22284. '']));
  22285. end;
  22286. procedure TTestModule.TestExtClassHelper_ClassMethod_MissingStatic;
  22287. begin
  22288. StartProgram(false);
  22289. Add([
  22290. '{$modeswitch externalclass}',
  22291. 'type',
  22292. ' TExtA = class external name ''ExtObj''',
  22293. ' procedure Run(w: word = 10);',
  22294. ' end;',
  22295. ' THelper = class helper for TExtA',
  22296. ' class procedure Fly;',
  22297. ' end;',
  22298. 'class procedure THelper.Fly;',
  22299. 'begin end;',
  22300. 'begin',
  22301. '']);
  22302. SetExpectedPasResolverError(sHelperClassMethodForExtClassMustBeStatic,
  22303. nHelperClassMethodForExtClassMustBeStatic);
  22304. ConvertProgram;
  22305. end;
  22306. procedure TTestModule.TestRecordHelper_ClassVar;
  22307. begin
  22308. StartProgram(false);
  22309. Add([
  22310. 'type',
  22311. ' TRec = record',
  22312. ' end;',
  22313. ' THelper = record helper for TRec',
  22314. ' const',
  22315. ' One = 1;',
  22316. ' Two: word = 2;',
  22317. ' class var',
  22318. ' Glob: word;',
  22319. ' function Foo(w: word): word;',
  22320. ' class function Bar(w: word): word; static;',
  22321. ' end;',
  22322. 'function THelper.foo(w: word): word;',
  22323. 'begin',
  22324. ' Result:=w;',
  22325. ' Two:=One+w;',
  22326. ' Glob:=Glob;',
  22327. ' Result:=Self.Glob;',
  22328. ' Self.Glob:=Self.Glob;',
  22329. ' with Self do Glob:=Glob;',
  22330. ' Self:=Self;',
  22331. 'end;',
  22332. 'class function THelper.bar(w: word): word;',
  22333. 'begin',
  22334. ' Result:=w;',
  22335. ' Two:=One;',
  22336. ' Glob:=Glob;',
  22337. 'end;',
  22338. 'var r: TRec;',
  22339. 'begin',
  22340. ' trec.two:=trec.one;',
  22341. ' trec.Glob:=trec.Glob;',
  22342. ' with trec do begin',
  22343. ' two:=one;',
  22344. ' Glob:=Glob;',
  22345. ' end;',
  22346. ' r.two:=r.one;',
  22347. ' r.Glob:=r.Glob;',
  22348. ' with r do begin',
  22349. ' two:=one;',
  22350. ' Glob:=Glob;',
  22351. ' end;',
  22352. '']);
  22353. ConvertProgram;
  22354. CheckSource('TestRecordHelper_ClassVar',
  22355. LinesToStr([ // statements
  22356. 'rtl.recNewT(this, "TRec", function () {',
  22357. ' this.$eq = function (b) {',
  22358. ' return true;',
  22359. ' };',
  22360. ' this.$assign = function (s) {',
  22361. ' return this;',
  22362. ' };',
  22363. '});',
  22364. 'rtl.createHelper(this, "THelper", null, function () {',
  22365. ' this.One = 1;',
  22366. ' this.Two = 2;',
  22367. ' this.Glob = 0;',
  22368. ' this.Foo = function (w) {',
  22369. ' var Result = 0;',
  22370. ' Result = w;',
  22371. ' $mod.THelper.Two = 1 + w;',
  22372. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22373. ' Result = $mod.THelper.Glob;',
  22374. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22375. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22376. ' this.$assign(this);',
  22377. ' return Result;',
  22378. ' };',
  22379. ' this.Bar = function (w) {',
  22380. ' var Result = 0;',
  22381. ' Result = w;',
  22382. ' $mod.THelper.Two = 1;',
  22383. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22384. ' return Result;',
  22385. ' };',
  22386. '});',
  22387. 'this.r = this.TRec.$new();',
  22388. '']),
  22389. LinesToStr([ // $mod.$main
  22390. '$mod.THelper.Two = 1;',
  22391. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22392. 'var $with = $mod.TRec;',
  22393. '$mod.THelper.Two = 1;',
  22394. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22395. '$mod.THelper.Two = 1;',
  22396. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22397. 'var $with1 = $mod.r;',
  22398. '$mod.THelper.Two = 1;',
  22399. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22400. '']));
  22401. end;
  22402. procedure TTestModule.TestRecordHelper_Method_Call;
  22403. begin
  22404. StartProgram(false);
  22405. Add([
  22406. '{$modeswitch AdvancedRecords}',
  22407. 'type',
  22408. ' TRec = record',
  22409. ' procedure Run(w: word = 10);',
  22410. ' end;',
  22411. ' THelper = record helper for TRec',
  22412. ' function Foo(w: word = 1): word;',
  22413. ' end;',
  22414. 'procedure TRec.Run(w: word);',
  22415. 'begin',
  22416. ' Foo;',
  22417. ' Foo();',
  22418. ' Foo(2);',
  22419. ' Self.Foo;',
  22420. ' Self.Foo();',
  22421. ' Self.Foo(3);',
  22422. ' with Self do begin',
  22423. ' Foo;',
  22424. ' Foo();',
  22425. ' Foo(4);',
  22426. ' end;',
  22427. 'end;',
  22428. 'function THelper.foo(w: word): word;',
  22429. 'begin',
  22430. ' Run;',
  22431. ' Run();',
  22432. ' Run(11);',
  22433. ' Foo;',
  22434. ' Foo();',
  22435. ' Foo(12);',
  22436. ' Self.Foo;',
  22437. ' Self.Foo();',
  22438. ' Self.Foo(13);',
  22439. ' with Self do begin',
  22440. ' Foo;',
  22441. ' Foo();',
  22442. ' Foo(14);',
  22443. ' end;',
  22444. 'end;',
  22445. 'var Rec: TRec;',
  22446. 'begin',
  22447. ' Rec.Foo;',
  22448. ' Rec.Foo();',
  22449. ' Rec.Foo(21);',
  22450. ' with Rec do begin',
  22451. ' Foo;',
  22452. ' Foo();',
  22453. ' Foo(22);',
  22454. ' end;',
  22455. '']);
  22456. ConvertProgram;
  22457. CheckSource('TestRecordHelper_Method_Call',
  22458. LinesToStr([ // statements
  22459. 'rtl.recNewT(this, "TRec", function () {',
  22460. ' this.$eq = function (b) {',
  22461. ' return true;',
  22462. ' };',
  22463. ' this.$assign = function (s) {',
  22464. ' return this;',
  22465. ' };',
  22466. ' this.Run = function (w) {',
  22467. ' $mod.THelper.Foo.call(this, 1);',
  22468. ' $mod.THelper.Foo.call(this, 1);',
  22469. ' $mod.THelper.Foo.call(this, 2);',
  22470. ' $mod.THelper.Foo.call(this, 1);',
  22471. ' $mod.THelper.Foo.call(this, 1);',
  22472. ' $mod.THelper.Foo.call(this, 3);',
  22473. ' $mod.THelper.Foo.call(this, 1);',
  22474. ' $mod.THelper.Foo.call(this, 1);',
  22475. ' $mod.THelper.Foo.call(this, 4);',
  22476. ' };',
  22477. '});',
  22478. 'rtl.createHelper(this, "THelper", null, function () {',
  22479. ' this.Foo = function (w) {',
  22480. ' var Result = 0;',
  22481. ' this.Run(10);',
  22482. ' this.Run(10);',
  22483. ' this.Run(11);',
  22484. ' $mod.THelper.Foo.call(this, 1);',
  22485. ' $mod.THelper.Foo.call(this, 1);',
  22486. ' $mod.THelper.Foo.call(this, 12);',
  22487. ' $mod.THelper.Foo.call(this, 1);',
  22488. ' $mod.THelper.Foo.call(this, 1);',
  22489. ' $mod.THelper.Foo.call(this, 13);',
  22490. ' $mod.THelper.Foo.call(this, 1);',
  22491. ' $mod.THelper.Foo.call(this, 1);',
  22492. ' $mod.THelper.Foo.call(this, 14);',
  22493. ' return Result;',
  22494. ' };',
  22495. '});',
  22496. 'this.Rec = this.TRec.$new();',
  22497. '']),
  22498. LinesToStr([ // $mod.$main
  22499. '$mod.THelper.Foo.call($mod.Rec, 1);',
  22500. '$mod.THelper.Foo.call($mod.Rec, 1);',
  22501. '$mod.THelper.Foo.call($mod.Rec, 21);',
  22502. 'var $with = $mod.Rec;',
  22503. '$mod.THelper.Foo.call($with, 1);',
  22504. '$mod.THelper.Foo.call($with, 1);',
  22505. '$mod.THelper.Foo.call($with, 22);',
  22506. '']));
  22507. end;
  22508. procedure TTestModule.TestRecordHelper_Constructor;
  22509. begin
  22510. StartProgram(false);
  22511. Add([
  22512. '{$modeswitch AdvancedRecords}',
  22513. 'type',
  22514. ' TRec = record',
  22515. ' constructor Create(w: word);',
  22516. ' end;',
  22517. ' THelper = record helper for TRec',
  22518. ' constructor NewHlp(w: word);',
  22519. ' end;',
  22520. 'var',
  22521. ' Rec: TRec;',
  22522. 'constructor TRec.Create(w: word);',
  22523. 'begin',
  22524. ' NewHlp(2);', // normal call
  22525. ' trec.NewHlp(3);', // new instance
  22526. 'end;',
  22527. 'constructor THelper.NewHlp(w: word);',
  22528. 'begin',
  22529. ' create(2);', // normal call
  22530. ' trec.create(3);', // new instance
  22531. ' NewHlp(4);', // normal call
  22532. ' trec.NewHlp(5);', // new instance
  22533. 'end;',
  22534. 'begin',
  22535. ' rec.newhlp(2);', // normal call
  22536. ' with rec do newhlp(12);', // normal call
  22537. ' trec.newhlp(3);', // new instance
  22538. ' with trec do newhlp(13);', // new instance
  22539. '']);
  22540. ConvertProgram;
  22541. CheckSource('TestRecordHelper_Constructor',
  22542. LinesToStr([ // statements
  22543. 'rtl.recNewT(this, "TRec", function () {',
  22544. ' this.$eq = function (b) {',
  22545. ' return true;',
  22546. ' };',
  22547. ' this.$assign = function (s) {',
  22548. ' return this;',
  22549. ' };',
  22550. ' this.Create = function (w) {',
  22551. ' $mod.THelper.NewHlp.call(this, 2);',
  22552. ' $mod.THelper.$new("NewHlp", [3]);',
  22553. ' return this;',
  22554. ' };',
  22555. '}, true);',
  22556. 'rtl.createHelper(this, "THelper", null, function () {',
  22557. ' this.NewHlp = function (w) {',
  22558. ' this.Create(2);',
  22559. ' $mod.TRec.$new().Create(3);',
  22560. ' $mod.THelper.NewHlp.call(this, 4);',
  22561. ' $mod.THelper.$new("NewHlp", [5]);',
  22562. ' return this;',
  22563. ' };',
  22564. ' this.$new = function (fn, args) {',
  22565. ' return this[fn].apply($mod.TRec.$new(), args);',
  22566. ' };',
  22567. '});',
  22568. 'this.Rec = this.TRec.$new();',
  22569. '']),
  22570. LinesToStr([ // $mod.$main
  22571. '$mod.THelper.NewHlp.call($mod.Rec, 2);',
  22572. 'var $with = $mod.Rec;',
  22573. '$mod.THelper.NewHlp.call($with, 12);',
  22574. '$mod.THelper.$new("NewHlp", [3]);',
  22575. 'var $with1 = $mod.TRec;',
  22576. '$mod.THelper.$new("NewHlp", [13]);',
  22577. '']));
  22578. end;
  22579. procedure TTestModule.TestTypeHelper_ClassVar;
  22580. begin
  22581. StartProgram(false);
  22582. Add([
  22583. '{$modeswitch typehelpers}',
  22584. 'type',
  22585. ' THelper = type helper for byte',
  22586. ' const',
  22587. ' One = 1;',
  22588. ' Two: word = 2;',
  22589. ' class var',
  22590. ' Glob: word;',
  22591. ' function Foo(w: word): word;',
  22592. ' class function Bar(w: word): word; static;',
  22593. ' end;',
  22594. 'function THelper.foo(w: word): word;',
  22595. 'begin',
  22596. ' Result:=w;',
  22597. ' Two:=One+w;',
  22598. ' Glob:=Glob;',
  22599. ' Result:=Self.Glob;',
  22600. ' Self.Glob:=Self.Glob;',
  22601. ' with Self do Glob:=Glob;',
  22602. 'end;',
  22603. 'class function THelper.bar(w: word): word;',
  22604. 'begin',
  22605. ' Result:=w;',
  22606. ' Two:=One;',
  22607. ' Glob:=Glob;',
  22608. 'end;',
  22609. 'var b: byte;',
  22610. 'begin',
  22611. ' byte.two:=byte.one;',
  22612. ' byte.Glob:=byte.Glob;',
  22613. ' with byte do begin',
  22614. ' two:=one;',
  22615. ' Glob:=Glob;',
  22616. ' end;',
  22617. ' b.two:=b.one;',
  22618. ' b.Glob:=b.Glob;',
  22619. ' with b do begin',
  22620. ' two:=one;',
  22621. ' Glob:=Glob;',
  22622. ' end;',
  22623. '']);
  22624. ConvertProgram;
  22625. CheckSource('TestTypeHelper_ClassVar',
  22626. LinesToStr([ // statements
  22627. 'rtl.createHelper(this, "THelper", null, function () {',
  22628. ' this.One = 1;',
  22629. ' this.Two = 2;',
  22630. ' this.Glob = 0;',
  22631. ' this.Foo = function (w) {',
  22632. ' var Result = 0;',
  22633. ' Result = w;',
  22634. ' $mod.THelper.Two = 1 + w;',
  22635. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22636. ' Result = $mod.THelper.Glob;',
  22637. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22638. ' var $with = this.get();',
  22639. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22640. ' return Result;',
  22641. ' };',
  22642. ' this.Bar = function (w) {',
  22643. ' var Result = 0;',
  22644. ' Result = w;',
  22645. ' $mod.THelper.Two = 1;',
  22646. ' $mod.THelper.Glob = $mod.THelper.Glob;',
  22647. ' return Result;',
  22648. ' };',
  22649. '});',
  22650. 'this.b = 0;',
  22651. '']),
  22652. LinesToStr([ // $mod.$main
  22653. '$mod.THelper.Two = 1;',
  22654. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22655. '$mod.THelper.Two = 1;',
  22656. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22657. '$mod.THelper.Two = 1;',
  22658. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22659. 'var $with = $mod.b;',
  22660. '$mod.THelper.Two = 1;',
  22661. '$mod.THelper.Glob = $mod.THelper.Glob;',
  22662. '']));
  22663. end;
  22664. procedure TTestModule.TestTypeHelper_PassResultElement;
  22665. begin
  22666. StartProgram(false);
  22667. Add([
  22668. '{$modeswitch typehelpers}',
  22669. 'type',
  22670. ' THelper = type helper for word',
  22671. ' procedure DoIt(e: byte = 123);',
  22672. ' class procedure DoSome(e: byte = 456); static;',
  22673. ' end;',
  22674. 'procedure THelper.DoIt(e: byte);',
  22675. 'begin',
  22676. 'end;',
  22677. 'class procedure THelper.DoSome(e: byte);',
  22678. 'begin',
  22679. 'end;',
  22680. 'function Foo(w: word): word;',
  22681. 'begin',
  22682. ' Result.DoIt;',
  22683. ' Result.DoIt();',
  22684. ' Result.DoSome;',
  22685. ' Result.DoSome();',
  22686. ' with Result do begin',
  22687. ' DoIt;',
  22688. ' DoIt();',
  22689. ' DoSome;',
  22690. ' DoSome();',
  22691. ' end;',
  22692. 'end;',
  22693. 'begin',
  22694. '']);
  22695. ConvertProgram;
  22696. CheckSource('TestTypeHelper_PassResultElement',
  22697. LinesToStr([ // statements
  22698. 'rtl.createHelper(this, "THelper", null, function () {',
  22699. ' this.DoIt = function (e) {',
  22700. ' };',
  22701. ' this.DoSome = function (e) {',
  22702. ' };',
  22703. '});',
  22704. 'this.Foo = function (w) {',
  22705. ' var Result = 0;',
  22706. ' $mod.THelper.DoIt.call({',
  22707. ' get: function () {',
  22708. ' return Result;',
  22709. ' },',
  22710. ' set: function (v) {',
  22711. ' Result = v;',
  22712. ' }',
  22713. ' }, 123);',
  22714. ' $mod.THelper.DoIt.call({',
  22715. ' get: function () {',
  22716. ' return Result;',
  22717. ' },',
  22718. ' set: function (v) {',
  22719. ' Result = v;',
  22720. ' }',
  22721. ' }, 123);',
  22722. ' $mod.THelper.DoSome(456);',
  22723. ' $mod.THelper.DoSome(456);',
  22724. ' $mod.THelper.DoIt.call({',
  22725. ' get: function () {',
  22726. ' return Result;',
  22727. ' },',
  22728. ' set: function (v) {',
  22729. ' Result = v;',
  22730. ' }',
  22731. ' }, 123);',
  22732. ' $mod.THelper.DoIt.call({',
  22733. ' get: function () {',
  22734. ' return Result;',
  22735. ' },',
  22736. ' set: function (v) {',
  22737. ' Result = v;',
  22738. ' }',
  22739. ' }, 123);',
  22740. ' $mod.THelper.DoSome(456);',
  22741. ' $mod.THelper.DoSome(456);',
  22742. ' return Result;',
  22743. '};',
  22744. '']),
  22745. LinesToStr([ // $mod.$main
  22746. '']));
  22747. end;
  22748. procedure TTestModule.TestTypeHelper_PassArgs;
  22749. begin
  22750. StartProgram(false);
  22751. Add([
  22752. '{$modeswitch typehelpers}',
  22753. 'type',
  22754. ' THelper = type helper for word',
  22755. ' procedure DoIt(e: byte = 123);',
  22756. ' end;',
  22757. 'procedure THelper.DoIt(e: byte);',
  22758. 'begin',
  22759. 'end;',
  22760. 'procedure FooDefault(a: word);',
  22761. 'begin',
  22762. ' a.DoIt;',
  22763. ' with a do DoIt;',
  22764. 'end;',
  22765. 'procedure FooConst(const a: word);',
  22766. 'begin',
  22767. ' a.DoIt;',
  22768. ' with a do DoIt;',
  22769. 'end;',
  22770. 'procedure FooVar(var a: word);',
  22771. 'begin',
  22772. ' a.DoIt;',
  22773. ' with a do DoIt;',
  22774. 'end;',
  22775. 'begin',
  22776. '']);
  22777. ConvertProgram;
  22778. CheckSource('TestTypeHelper_PassArgs',
  22779. LinesToStr([ // statements
  22780. 'rtl.createHelper(this, "THelper", null, function () {',
  22781. ' this.DoIt = function (e) {',
  22782. ' };',
  22783. '});',
  22784. 'this.FooDefault = function (a) {',
  22785. ' $mod.THelper.DoIt.call({',
  22786. ' get: function () {',
  22787. ' return a;',
  22788. ' },',
  22789. ' set: function (v) {',
  22790. ' a = v;',
  22791. ' }',
  22792. ' }, 123);',
  22793. ' $mod.THelper.DoIt.call({',
  22794. ' get: function () {',
  22795. ' return a;',
  22796. ' },',
  22797. ' set: function (v) {',
  22798. ' a = v;',
  22799. ' }',
  22800. ' }, 123);',
  22801. '};',
  22802. 'this.FooConst = function (a) {',
  22803. ' $mod.THelper.DoIt.call({',
  22804. ' get: function () {',
  22805. ' return a;',
  22806. ' },',
  22807. ' set: function (v) {',
  22808. ' rtl.raiseE("EPropReadOnly");',
  22809. ' }',
  22810. ' }, 123);',
  22811. ' $mod.THelper.DoIt.call({',
  22812. ' get: function () {',
  22813. ' return a;',
  22814. ' },',
  22815. ' set: function () {',
  22816. ' rtl.raiseE("EPropReadOnly");',
  22817. ' }',
  22818. ' }, 123);',
  22819. '};',
  22820. 'this.FooVar = function (a) {',
  22821. ' $mod.THelper.DoIt.call(a, 123);',
  22822. ' var $with = a.get();',
  22823. ' $mod.THelper.DoIt.call(a, 123);',
  22824. '};',
  22825. '']),
  22826. LinesToStr([ // $mod.$main
  22827. '']));
  22828. end;
  22829. procedure TTestModule.TestTypeHelper_PassVarConst;
  22830. begin
  22831. StartProgram(false);
  22832. Add([
  22833. '{$modeswitch typehelpers}',
  22834. 'type',
  22835. ' THelper = type helper for word',
  22836. ' procedure DoIt(e: byte = 123);',
  22837. ' end;',
  22838. 'procedure THelper.DoIt(e: byte);',
  22839. 'begin',
  22840. 'end;',
  22841. 'var a: word;',
  22842. 'const c: word = 2;',
  22843. '{$writeableconst off}',
  22844. 'const r: word = 3;',
  22845. 'begin',
  22846. ' a.DoIt;',
  22847. ' with a do DoIt;',
  22848. ' c.DoIt;',
  22849. ' with c do DoIt;',
  22850. ' r.DoIt;',
  22851. ' with r do DoIt;',
  22852. '']);
  22853. ConvertProgram;
  22854. CheckSource('TestTypeHelper_PassVarConst',
  22855. LinesToStr([ // statements
  22856. 'rtl.createHelper(this, "THelper", null, function () {',
  22857. ' this.DoIt = function (e) {',
  22858. ' };',
  22859. '});',
  22860. 'this.a = 0;',
  22861. 'this.c = 2;',
  22862. 'this.r = 3;',
  22863. '']),
  22864. LinesToStr([ // $mod.$main
  22865. '$mod.THelper.DoIt.call({',
  22866. ' p: $mod,',
  22867. ' get: function () {',
  22868. ' return this.p.a;',
  22869. ' },',
  22870. ' set: function (v) {',
  22871. ' this.p.a = v;',
  22872. ' }',
  22873. '}, 123);',
  22874. 'var $with = $mod.a;',
  22875. '$mod.THelper.DoIt.call({',
  22876. ' get: function () {',
  22877. ' return $with;',
  22878. ' },',
  22879. ' set: function (v) {',
  22880. ' $with = v;',
  22881. ' }',
  22882. '}, 123);',
  22883. '$mod.THelper.DoIt.call({',
  22884. ' p: $mod,',
  22885. ' get: function () {',
  22886. ' return this.p.c;',
  22887. ' },',
  22888. ' set: function (v) {',
  22889. ' this.p.c = v;',
  22890. ' }',
  22891. '}, 123);',
  22892. 'var $with1 = $mod.c;',
  22893. '$mod.THelper.DoIt.call({',
  22894. ' get: function () {',
  22895. ' return $with1;',
  22896. ' },',
  22897. ' set: function (v) {',
  22898. ' $with1 = v;',
  22899. ' }',
  22900. '}, 123);',
  22901. '$mod.THelper.DoIt.call({',
  22902. ' get: function () {',
  22903. ' return 3;',
  22904. ' },',
  22905. ' set: function (v) {',
  22906. ' rtl.raiseE("EPropReadOnly");',
  22907. ' }',
  22908. '}, 123);',
  22909. 'var $with2 = 3;',
  22910. ' $mod.THelper.DoIt.call({',
  22911. ' get: function () {',
  22912. ' return $with2;',
  22913. ' },',
  22914. ' set: function () {',
  22915. ' rtl.raiseE("EPropReadOnly");',
  22916. ' }',
  22917. ' }, 123);',
  22918. '']));
  22919. end;
  22920. procedure TTestModule.TestTypeHelper_PassFuncResult;
  22921. begin
  22922. StartProgram(false);
  22923. Add([
  22924. '{$modeswitch typehelpers}',
  22925. 'type',
  22926. ' THelper = type helper for word',
  22927. ' procedure DoIt(e: byte = 123);',
  22928. ' end;',
  22929. 'procedure THelper.DoIt(e: byte);',
  22930. 'begin',
  22931. 'end;',
  22932. 'function Foo(b: byte = 1): word;',
  22933. 'begin',
  22934. 'end;',
  22935. 'begin',
  22936. ' Foo.DoIt;',
  22937. ' Foo().DoIt;',
  22938. ' with Foo do DoIt;',
  22939. ' with Foo() do DoIt;',
  22940. '']);
  22941. ConvertProgram;
  22942. CheckSource('TestTypeHelper_PassFuncResult',
  22943. LinesToStr([ // statements
  22944. 'rtl.createHelper(this, "THelper", null, function () {',
  22945. ' this.DoIt = function (e) {',
  22946. ' };',
  22947. '});',
  22948. 'this.Foo = function (b) {',
  22949. ' var Result = 0;',
  22950. ' return Result;',
  22951. '};',
  22952. '']),
  22953. LinesToStr([ // $mod.$main
  22954. '$mod.THelper.DoIt.call({',
  22955. ' a: $mod.Foo(1),',
  22956. ' get: function () {',
  22957. ' return this.a;',
  22958. ' },',
  22959. ' set: function (v) {',
  22960. ' this.a = v;',
  22961. ' }',
  22962. '}, 123);',
  22963. '$mod.THelper.DoIt.call({',
  22964. ' a: $mod.Foo(1),',
  22965. ' get: function () {',
  22966. ' return this.a;',
  22967. ' },',
  22968. ' set: function (v) {',
  22969. ' this.a = v;',
  22970. ' }',
  22971. '}, 123);',
  22972. 'var $with = $mod.Foo(1);',
  22973. '$mod.THelper.DoIt.call({',
  22974. ' get: function () {',
  22975. ' return $with;',
  22976. ' },',
  22977. ' set: function (v) {',
  22978. ' $with = v;',
  22979. ' }',
  22980. '}, 123);',
  22981. 'var $with1 = $mod.Foo(1);',
  22982. '$mod.THelper.DoIt.call({',
  22983. ' get: function () {',
  22984. ' return $with1;',
  22985. ' },',
  22986. ' set: function (v) {',
  22987. ' $with1 = v;',
  22988. ' }',
  22989. '}, 123);',
  22990. '']));
  22991. end;
  22992. procedure TTestModule.TestTypeHelper_PassPropertyField;
  22993. begin
  22994. StartProgram(false);
  22995. Add([
  22996. '{$modeswitch typehelpers}',
  22997. 'type',
  22998. ' TObject = class',
  22999. ' FField: word;',
  23000. ' procedure SetField(Value: word);',
  23001. ' property Field: word read FField write SetField;',
  23002. ' end;',
  23003. ' THelper = type helper for word',
  23004. ' procedure Fly;',
  23005. ' class procedure Run; static;',
  23006. ' end;',
  23007. 'procedure TObject.SetField(Value: word);',
  23008. 'begin',
  23009. ' Field.Fly;',
  23010. ' Field.Run;',
  23011. ' Self.Field.Fly;',
  23012. ' Self.Field.Run;',
  23013. ' with Self do begin',
  23014. ' Field.Fly;',
  23015. ' Field.Run;',
  23016. ' end;',
  23017. ' with Self.Field do begin',
  23018. ' Fly;',
  23019. ' Run;',
  23020. ' end;',
  23021. 'end;',
  23022. 'procedure THelper.Fly;',
  23023. 'begin',
  23024. 'end;',
  23025. 'class procedure THelper.Run;',
  23026. 'begin',
  23027. 'end;',
  23028. 'var',
  23029. ' o: TObject;',
  23030. 'begin',
  23031. ' o.Field.Fly;',
  23032. ' o.Field.Run;',
  23033. ' with o do begin',
  23034. ' Field.Fly;',
  23035. ' Field.Run;',
  23036. ' end;',
  23037. ' with o.Field do begin',
  23038. ' Fly;',
  23039. ' Run;',
  23040. ' end;',
  23041. '']);
  23042. ConvertProgram;
  23043. CheckSource('TestTypeHelper_PassPropertyField',
  23044. LinesToStr([ // statements
  23045. 'rtl.createClass(this, "TObject", null, function () {',
  23046. ' this.$init = function () {',
  23047. ' this.FField = 0;',
  23048. ' };',
  23049. ' this.$final = function () {',
  23050. ' };',
  23051. ' this.SetField = function (Value) {',
  23052. ' $mod.THelper.Fly.call({',
  23053. ' p: this,',
  23054. ' get: function () {',
  23055. ' return this.p.FField;',
  23056. ' },',
  23057. ' set: function (v) {',
  23058. ' this.p.FField = v;',
  23059. ' }',
  23060. ' });',
  23061. ' $mod.THelper.Run();',
  23062. ' $mod.THelper.Fly.call({',
  23063. ' p: this,',
  23064. ' get: function () {',
  23065. ' return this.p.FField;',
  23066. ' },',
  23067. ' set: function (v) {',
  23068. ' this.p.FField = v;',
  23069. ' }',
  23070. ' });',
  23071. ' $mod.THelper.Run();',
  23072. ' $mod.THelper.Fly.call({',
  23073. ' p: this,',
  23074. ' get: function () {',
  23075. ' return this.p.FField;',
  23076. ' },',
  23077. ' set: function (v) {',
  23078. ' this.p.FField = v;',
  23079. ' }',
  23080. ' });',
  23081. ' $mod.THelper.Run();',
  23082. ' var $with = this.FField;',
  23083. ' $mod.THelper.Fly.call({',
  23084. ' get: function () {',
  23085. ' return $with;',
  23086. ' },',
  23087. ' set: function (v) {',
  23088. ' $with = v;',
  23089. ' }',
  23090. ' });',
  23091. ' $mod.THelper.Run();',
  23092. ' };',
  23093. '});',
  23094. 'rtl.createHelper(this, "THelper", null, function () {',
  23095. ' this.Fly = function () {',
  23096. ' };',
  23097. ' this.Run = function () {',
  23098. ' };',
  23099. '});',
  23100. 'this.o = null;',
  23101. '']),
  23102. LinesToStr([ // $mod.$main
  23103. '$mod.THelper.Fly.call({',
  23104. ' p: $mod.o,',
  23105. ' get: function () {',
  23106. ' return this.p.FField;',
  23107. ' },',
  23108. ' set: function (v) {',
  23109. ' this.p.FField = v;',
  23110. ' }',
  23111. '});',
  23112. '$mod.THelper.Run();',
  23113. 'var $with = $mod.o;',
  23114. '$mod.THelper.Fly.call({',
  23115. ' p: $with,',
  23116. ' get: function () {',
  23117. ' return this.p.FField;',
  23118. ' },',
  23119. ' set: function (v) {',
  23120. ' this.p.FField = v;',
  23121. ' }',
  23122. '});',
  23123. '$mod.THelper.Run();',
  23124. 'var $with1 = $mod.o.FField;',
  23125. '$mod.THelper.Fly.call({',
  23126. ' get: function () {',
  23127. ' return $with1;',
  23128. ' },',
  23129. ' set: function (v) {',
  23130. ' $with1 = v;',
  23131. ' }',
  23132. '});',
  23133. '$mod.THelper.Run();',
  23134. '']));
  23135. end;
  23136. procedure TTestModule.TestTypeHelper_PassPropertyGetter;
  23137. begin
  23138. StartProgram(false);
  23139. Add([
  23140. '{$modeswitch typehelpers}',
  23141. 'type',
  23142. ' TObject = class',
  23143. ' FField: word;',
  23144. ' function GetField: word;',
  23145. ' property Field: word read GetField write FField;',
  23146. ' end;',
  23147. ' THelper = type helper for word',
  23148. ' procedure Fly;',
  23149. ' class procedure Run; static;',
  23150. ' end;',
  23151. 'function TObject.GetField: word;',
  23152. 'begin',
  23153. ' Field.Fly;',
  23154. ' Field.Run;',
  23155. ' Self.Field.Fly;',
  23156. ' Self.Field.Run;',
  23157. ' with Self do begin',
  23158. ' Field.Fly;',
  23159. ' Field.Run;',
  23160. ' end;',
  23161. ' with Self.Field do begin',
  23162. ' Fly;',
  23163. ' Run;',
  23164. ' end;',
  23165. 'end;',
  23166. 'procedure THelper.Fly;',
  23167. 'begin',
  23168. 'end;',
  23169. 'class procedure THelper.Run;',
  23170. 'begin',
  23171. 'end;',
  23172. 'var',
  23173. ' o: TObject;',
  23174. 'begin',
  23175. ' o.Field.Fly;',
  23176. ' o.Field.Run;',
  23177. ' with o do begin',
  23178. ' Field.Fly;',
  23179. ' Field.Run;',
  23180. ' end;',
  23181. ' with o.Field do begin',
  23182. ' Fly;',
  23183. ' Run;',
  23184. ' end;',
  23185. '']);
  23186. ConvertProgram;
  23187. CheckSource('TestTypeHelper_PassPropertyGetter',
  23188. LinesToStr([ // statements
  23189. 'rtl.createClass(this, "TObject", null, function () {',
  23190. ' this.$init = function () {',
  23191. ' this.FField = 0;',
  23192. ' };',
  23193. ' this.$final = function () {',
  23194. ' };',
  23195. ' this.GetField = function () {',
  23196. ' var Result = 0;',
  23197. ' $mod.THelper.Fly.call({',
  23198. ' p: this.GetField(),',
  23199. ' get: function () {',
  23200. ' return this.p;',
  23201. ' },',
  23202. ' set: function (v) {',
  23203. ' this.p = v;',
  23204. ' }',
  23205. ' });',
  23206. ' $mod.THelper.Run();',
  23207. ' $mod.THelper.Fly.call({',
  23208. ' p: this.GetField(),',
  23209. ' get: function () {',
  23210. ' return this.p;',
  23211. ' },',
  23212. ' set: function (v) {',
  23213. ' this.p = v;',
  23214. ' }',
  23215. ' });',
  23216. ' $mod.THelper.Run();',
  23217. ' $mod.THelper.Fly.call({',
  23218. ' p: this.GetField(),',
  23219. ' get: function () {',
  23220. ' return this.p;',
  23221. ' },',
  23222. ' set: function (v) {',
  23223. ' this.p = v;',
  23224. ' }',
  23225. ' });',
  23226. ' $mod.THelper.Run();',
  23227. ' var $with = this.GetField();',
  23228. ' $mod.THelper.Fly.call({',
  23229. ' get: function () {',
  23230. ' return $with;',
  23231. ' },',
  23232. ' set: function (v) {',
  23233. ' $with = v;',
  23234. ' }',
  23235. ' });',
  23236. ' $mod.THelper.Run();',
  23237. ' return Result;',
  23238. ' };',
  23239. '});',
  23240. 'rtl.createHelper(this, "THelper", null, function () {',
  23241. ' this.Fly = function () {',
  23242. ' };',
  23243. ' this.Run = function () {',
  23244. ' };',
  23245. '});',
  23246. 'this.o = null;',
  23247. '']),
  23248. LinesToStr([ // $mod.$main
  23249. '$mod.THelper.Fly.call({',
  23250. ' p: $mod.o.GetField(),',
  23251. ' get: function () {',
  23252. ' return this.p;',
  23253. ' },',
  23254. ' set: function (v) {',
  23255. ' this.p = v;',
  23256. ' }',
  23257. '});',
  23258. '$mod.THelper.Run();',
  23259. 'var $with = $mod.o;',
  23260. '$mod.THelper.Fly.call({',
  23261. ' p: $with.GetField(),',
  23262. ' get: function () {',
  23263. ' return this.p;',
  23264. ' },',
  23265. ' set: function (v) {',
  23266. ' this.p = v;',
  23267. ' }',
  23268. '});',
  23269. '$mod.THelper.Run();',
  23270. 'var $with1 = $mod.o.GetField();',
  23271. '$mod.THelper.Fly.call({',
  23272. ' get: function () {',
  23273. ' return $with1;',
  23274. ' },',
  23275. ' set: function (v) {',
  23276. ' $with1 = v;',
  23277. ' }',
  23278. '});',
  23279. '$mod.THelper.Run();',
  23280. '']));
  23281. end;
  23282. procedure TTestModule.TestTypeHelper_PassClassPropertyField;
  23283. begin
  23284. StartProgram(false);
  23285. Add([
  23286. '{$modeswitch typehelpers}',
  23287. 'type',
  23288. ' TObject = class',
  23289. ' class var FField: word;',
  23290. ' class procedure SetField(Value: word);',
  23291. ' class property Field: word read FField write SetField;',
  23292. ' end;',
  23293. ' THelper = type helper for word',
  23294. ' procedure Fly(n: byte);',
  23295. ' end;',
  23296. 'class procedure TObject.SetField(Value: word);',
  23297. 'begin',
  23298. ' Field.Fly(1);',
  23299. ' Self.Field.Fly(2);',
  23300. ' with Self do Field.Fly(3);',
  23301. ' with Self.Field do Fly(4);',
  23302. ' TObject.Field.Fly(5);',
  23303. ' with TObject do Field.Fly(6);',
  23304. ' with TObject.Field do Fly(7);',
  23305. 'end;',
  23306. 'procedure THelper.Fly(n: byte);',
  23307. 'begin',
  23308. 'end;',
  23309. 'var',
  23310. ' o: TObject;',
  23311. 'begin',
  23312. ' o.Field.Fly(11);',
  23313. ' with o do Field.Fly(12);',
  23314. ' with o.Field do Fly(13);',
  23315. ' TObject.Field.Fly(14);',
  23316. ' with TObject do Field.Fly(15);',
  23317. ' with TObject.Field do Fly(16);',
  23318. '']);
  23319. ConvertProgram;
  23320. CheckSource('TestTypeHelper_PassClassPropertyField',
  23321. LinesToStr([ // statements
  23322. 'rtl.createClass(this, "TObject", null, function () {',
  23323. ' this.FField = 0;',
  23324. ' this.$init = function () {',
  23325. ' };',
  23326. ' this.$final = function () {',
  23327. ' };',
  23328. ' this.SetField = function (Value) {',
  23329. ' $mod.THelper.Fly.call({',
  23330. ' p: this,',
  23331. ' get: function () {',
  23332. ' return this.p.FField;',
  23333. ' },',
  23334. ' set: function (v) {',
  23335. ' $mod.TObject.FField = v;',
  23336. ' }',
  23337. ' }, 1);',
  23338. ' $mod.THelper.Fly.call({',
  23339. ' p: this,',
  23340. ' get: function () {',
  23341. ' return this.p.FField;',
  23342. ' },',
  23343. ' set: function (v) {',
  23344. ' $mod.TObject.FField = v;',
  23345. ' }',
  23346. ' }, 2);',
  23347. ' $mod.THelper.Fly.call({',
  23348. ' p: this,',
  23349. ' get: function () {',
  23350. ' return this.p.FField;',
  23351. ' },',
  23352. ' set: function (v) {',
  23353. ' $mod.TObject.FField = v;',
  23354. ' }',
  23355. ' }, 3);',
  23356. ' var $with = this.FField;',
  23357. ' $mod.THelper.Fly.call({',
  23358. ' get: function () {',
  23359. ' return $with;',
  23360. ' },',
  23361. ' set: function (v) {',
  23362. ' $with = v;',
  23363. ' }',
  23364. ' }, 4);',
  23365. ' $mod.THelper.Fly.call({',
  23366. ' p: $mod.TObject,',
  23367. ' get: function () {',
  23368. ' return this.p.FField;',
  23369. ' },',
  23370. ' set: function (v) {',
  23371. ' $mod.TObject.FField = v;',
  23372. ' }',
  23373. ' }, 5);',
  23374. ' var $with1 = $mod.TObject;',
  23375. ' $mod.THelper.Fly.call({',
  23376. ' p: $with1,',
  23377. ' get: function () {',
  23378. ' return this.p.FField;',
  23379. ' },',
  23380. ' set: function (v) {',
  23381. ' $mod.TObject.FField = v;',
  23382. ' }',
  23383. ' }, 6);',
  23384. ' var $with2 = $mod.TObject.FField;',
  23385. ' $mod.THelper.Fly.call({',
  23386. ' get: function () {',
  23387. ' return $with2;',
  23388. ' },',
  23389. ' set: function (v) {',
  23390. ' $with2 = v;',
  23391. ' }',
  23392. ' }, 7);',
  23393. ' };',
  23394. '});',
  23395. 'rtl.createHelper(this, "THelper", null, function () {',
  23396. ' this.Fly = function (n) {',
  23397. ' };',
  23398. '});',
  23399. 'this.o = null;',
  23400. '']),
  23401. LinesToStr([ // $mod.$main
  23402. '$mod.THelper.Fly.call({',
  23403. ' p: $mod.o,',
  23404. ' get: function () {',
  23405. ' return this.p.FField;',
  23406. ' },',
  23407. ' set: function (v) {',
  23408. ' $mod.TObject.FField = v;',
  23409. ' }',
  23410. '}, 11);',
  23411. 'var $with = $mod.o;',
  23412. '$mod.THelper.Fly.call({',
  23413. ' p: $with,',
  23414. ' get: function () {',
  23415. ' return this.p.FField;',
  23416. ' },',
  23417. ' set: function (v) {',
  23418. ' $mod.TObject.FField = v;',
  23419. ' }',
  23420. '}, 12);',
  23421. 'var $with1 = $mod.o.FField;',
  23422. '$mod.THelper.Fly.call({',
  23423. ' get: function () {',
  23424. ' return $with1;',
  23425. ' },',
  23426. ' set: function (v) {',
  23427. ' $with1 = v;',
  23428. ' }',
  23429. '}, 13);',
  23430. '$mod.THelper.Fly.call({',
  23431. ' p: $mod.TObject,',
  23432. ' get: function () {',
  23433. ' return this.p.FField;',
  23434. ' },',
  23435. ' set: function (v) {',
  23436. ' $mod.TObject.FField = v;',
  23437. ' }',
  23438. '}, 14);',
  23439. 'var $with2 = $mod.TObject;',
  23440. '$mod.THelper.Fly.call({',
  23441. ' p: $with2,',
  23442. ' get: function () {',
  23443. ' return this.p.FField;',
  23444. ' },',
  23445. ' set: function (v) {',
  23446. ' $mod.TObject.FField = v;',
  23447. ' }',
  23448. '}, 15);',
  23449. 'var $with3 = $mod.TObject.FField;',
  23450. '$mod.THelper.Fly.call({',
  23451. ' get: function () {',
  23452. ' return $with3;',
  23453. ' },',
  23454. ' set: function (v) {',
  23455. ' $with3 = v;',
  23456. ' }',
  23457. '}, 16);',
  23458. '']));
  23459. end;
  23460. procedure TTestModule.TestTypeHelper_PassClassPropertyGetterStatic;
  23461. begin
  23462. StartProgram(false);
  23463. Add([
  23464. '{$modeswitch typehelpers}',
  23465. 'type',
  23466. ' TObject = class',
  23467. ' class var FField: word;',
  23468. ' class function GetField: word; static;',
  23469. ' class property Field: word read GetField write FField;',
  23470. ' end;',
  23471. ' THelper = type helper for word',
  23472. ' procedure Fly(n: byte);',
  23473. ' end;',
  23474. 'class function TObject.GetField: word;',
  23475. 'begin',
  23476. ' Field.Fly(1);',
  23477. ' TObject.Field.Fly(5);',
  23478. ' with TObject do Field.Fly(6);',
  23479. ' with TObject.Field do Fly(7);',
  23480. 'end;',
  23481. 'procedure THelper.Fly(n: byte);',
  23482. 'begin',
  23483. 'end;',
  23484. 'var',
  23485. ' o: TObject;',
  23486. 'begin',
  23487. ' o.Field.Fly(11);',
  23488. ' with o do Field.Fly(12);',
  23489. ' with o.Field do Fly(13);',
  23490. '']);
  23491. ConvertProgram;
  23492. CheckSource('TestTypeHelper_PassClassPropertyGetterStatic',
  23493. LinesToStr([ // statements
  23494. 'rtl.createClass(this, "TObject", null, function () {',
  23495. ' this.FField = 0;',
  23496. ' this.$init = function () {',
  23497. ' };',
  23498. ' this.$final = function () {',
  23499. ' };',
  23500. ' this.GetField = function () {',
  23501. ' var Result = 0;',
  23502. ' $mod.THelper.Fly.call({',
  23503. ' p: $mod.TObject.GetField(),',
  23504. ' get: function () {',
  23505. ' return this.p;',
  23506. ' },',
  23507. ' set: function (v) {',
  23508. ' this.p = v;',
  23509. ' }',
  23510. ' }, 1);',
  23511. ' $mod.THelper.Fly.call({',
  23512. ' p: $mod.TObject.GetField(),',
  23513. ' get: function () {',
  23514. ' return this.p;',
  23515. ' },',
  23516. ' set: function (v) {',
  23517. ' this.p = v;',
  23518. ' }',
  23519. ' }, 5);',
  23520. ' var $with = $mod.TObject;',
  23521. ' $mod.THelper.Fly.call({',
  23522. ' p: $with.GetField(),',
  23523. ' get: function () {',
  23524. ' return this.p;',
  23525. ' },',
  23526. ' set: function (v) {',
  23527. ' this.p = v;',
  23528. ' }',
  23529. ' }, 6);',
  23530. ' var $with1 = $mod.TObject.GetField();',
  23531. ' $mod.THelper.Fly.call({',
  23532. ' get: function () {',
  23533. ' return $with1;',
  23534. ' },',
  23535. ' set: function (v) {',
  23536. ' $with1 = v;',
  23537. ' }',
  23538. ' }, 7);',
  23539. ' return Result;',
  23540. ' };',
  23541. '});',
  23542. 'rtl.createHelper(this, "THelper", null, function () {',
  23543. ' this.Fly = function (n) {',
  23544. ' };',
  23545. '});',
  23546. 'this.o = null;',
  23547. '']),
  23548. LinesToStr([ // $mod.$main
  23549. '$mod.THelper.Fly.call({',
  23550. ' p: $mod.o.GetField(),',
  23551. ' get: function () {',
  23552. ' return this.p;',
  23553. ' },',
  23554. ' set: function (v) {',
  23555. ' this.p = v;',
  23556. ' }',
  23557. '}, 11);',
  23558. 'var $with = $mod.o;',
  23559. '$mod.THelper.Fly.call({',
  23560. ' p: $with.GetField(),',
  23561. ' get: function () {',
  23562. ' return this.p;',
  23563. ' },',
  23564. ' set: function (v) {',
  23565. ' this.p = v;',
  23566. ' }',
  23567. '}, 12);',
  23568. 'var $with1 = $mod.o.GetField();',
  23569. '$mod.THelper.Fly.call({',
  23570. ' get: function () {',
  23571. ' return $with1;',
  23572. ' },',
  23573. ' set: function (v) {',
  23574. ' $with1 = v;',
  23575. ' }',
  23576. '}, 13);',
  23577. '']));
  23578. end;
  23579. procedure TTestModule.TestTypeHelper_PassClassPropertyGetterNonStatic;
  23580. begin
  23581. StartProgram(false);
  23582. Add([
  23583. '{$modeswitch typehelpers}',
  23584. 'type',
  23585. ' TObject = class',
  23586. ' class var FField: word;',
  23587. ' class function GetField: word;',
  23588. ' class property Field: word read GetField write FField;',
  23589. ' end;',
  23590. ' TClass = class of TObject;',
  23591. ' THelper = type helper for word',
  23592. ' procedure Fly(n: byte);',
  23593. ' end;',
  23594. 'class function TObject.GetField: word;',
  23595. 'begin',
  23596. ' Field.Fly(1);',
  23597. ' Self.Field.Fly(5);',
  23598. ' with Self do Field.Fly(6);',
  23599. ' with Self.Field do Fly(7);',
  23600. 'end;',
  23601. 'procedure THelper.Fly(n: byte);',
  23602. 'begin',
  23603. 'end;',
  23604. 'var',
  23605. ' o: TObject;',
  23606. ' c: TClass;',
  23607. 'begin',
  23608. ' o.Field.Fly(11);',
  23609. ' with o do Field.Fly(12);',
  23610. ' with o.Field do Fly(13);',
  23611. ' c.Field.Fly(14);',
  23612. ' with c do Field.Fly(15);',
  23613. ' with c.Field do Fly(16);',
  23614. '']);
  23615. ConvertProgram;
  23616. CheckSource('TestTypeHelper_PassClassPropertyGetterNonStatic',
  23617. LinesToStr([ // statements
  23618. 'rtl.createClass(this, "TObject", null, function () {',
  23619. ' this.FField = 0;',
  23620. ' this.$init = function () {',
  23621. ' };',
  23622. ' this.$final = function () {',
  23623. ' };',
  23624. ' this.GetField = function () {',
  23625. ' var Result = 0;',
  23626. ' $mod.THelper.Fly.call({',
  23627. ' p: this.GetField(),',
  23628. ' get: function () {',
  23629. ' return this.p;',
  23630. ' },',
  23631. ' set: function (v) {',
  23632. ' this.p = v;',
  23633. ' }',
  23634. ' }, 1);',
  23635. ' $mod.THelper.Fly.call({',
  23636. ' p: this.GetField(),',
  23637. ' get: function () {',
  23638. ' return this.p;',
  23639. ' },',
  23640. ' set: function (v) {',
  23641. ' this.p = v;',
  23642. ' }',
  23643. ' }, 5);',
  23644. ' $mod.THelper.Fly.call({',
  23645. ' p: this.GetField(),',
  23646. ' get: function () {',
  23647. ' return this.p;',
  23648. ' },',
  23649. ' set: function (v) {',
  23650. ' this.p = v;',
  23651. ' }',
  23652. ' }, 6);',
  23653. ' var $with = this.GetField();',
  23654. ' $mod.THelper.Fly.call({',
  23655. ' get: function () {',
  23656. ' return $with;',
  23657. ' },',
  23658. ' set: function (v) {',
  23659. ' $with = v;',
  23660. ' }',
  23661. ' }, 7);',
  23662. ' return Result;',
  23663. ' };',
  23664. '});',
  23665. 'rtl.createHelper(this, "THelper", null, function () {',
  23666. ' this.Fly = function (n) {',
  23667. ' };',
  23668. '});',
  23669. 'this.o = null;',
  23670. 'this.c = null;',
  23671. '']),
  23672. LinesToStr([ // $mod.$main
  23673. '$mod.THelper.Fly.call({',
  23674. ' p: $mod.o.$class.GetField(),',
  23675. ' get: function () {',
  23676. ' return this.p;',
  23677. ' },',
  23678. ' set: function (v) {',
  23679. ' this.p = v;',
  23680. ' }',
  23681. '}, 11);',
  23682. 'var $with = $mod.o;',
  23683. '$mod.THelper.Fly.call({',
  23684. ' p: $with.$class.GetField(),',
  23685. ' get: function () {',
  23686. ' return this.p;',
  23687. ' },',
  23688. ' set: function (v) {',
  23689. ' this.p = v;',
  23690. ' }',
  23691. '}, 12);',
  23692. 'var $with1 = $mod.o.$class.GetField();',
  23693. '$mod.THelper.Fly.call({',
  23694. ' get: function () {',
  23695. ' return $with1;',
  23696. ' },',
  23697. ' set: function (v) {',
  23698. ' $with1 = v;',
  23699. ' }',
  23700. '}, 13);',
  23701. '$mod.THelper.Fly.call({',
  23702. ' p: $mod.c.GetField(),',
  23703. ' get: function () {',
  23704. ' return this.p;',
  23705. ' },',
  23706. ' set: function (v) {',
  23707. ' this.p = v;',
  23708. ' }',
  23709. '}, 14);',
  23710. 'var $with2 = $mod.c;',
  23711. '$mod.THelper.Fly.call({',
  23712. ' p: $with2.GetField(),',
  23713. ' get: function () {',
  23714. ' return this.p;',
  23715. ' },',
  23716. ' set: function (v) {',
  23717. ' this.p = v;',
  23718. ' }',
  23719. '}, 15);',
  23720. 'var $with3 = $mod.c.GetField();',
  23721. '$mod.THelper.Fly.call({',
  23722. ' get: function () {',
  23723. ' return $with3;',
  23724. ' },',
  23725. ' set: function (v) {',
  23726. ' $with3 = v;',
  23727. ' }',
  23728. '}, 16);',
  23729. '']));
  23730. end;
  23731. procedure TTestModule.TestTypeHelper_Property;
  23732. begin
  23733. StartProgram(false);
  23734. Add([
  23735. '{$modeswitch typehelpers}',
  23736. 'type',
  23737. ' THelper = type helper for word',
  23738. ' function GetSize: longint;',
  23739. ' procedure SetSize(Value: longint);',
  23740. ' property Size: longint read GetSize write SetSize;',
  23741. ' end;',
  23742. 'function THelper.GetSize: longint;',
  23743. 'begin',
  23744. ' Result:=Size+1;',
  23745. ' Size:=2;',
  23746. ' Result:=Self.Size+3;',
  23747. ' Self.Size:=4;',
  23748. ' with Self do begin',
  23749. ' Result:=Size+5;',
  23750. ' Size:=6;',
  23751. ' end;',
  23752. 'end;',
  23753. 'procedure THelper.SetSize(Value: longint);',
  23754. 'begin',
  23755. 'end;',
  23756. 'var w: word;',
  23757. 'begin',
  23758. ' w:=w.Size+7;',
  23759. ' w.Size:=w+8;',
  23760. ' with w do begin',
  23761. ' w:=Size+9;',
  23762. ' Size:=w+10;',
  23763. ' end;',
  23764. '']);
  23765. ConvertProgram;
  23766. CheckSource('TestTypeHelper_Property',
  23767. LinesToStr([ // statements
  23768. 'rtl.createHelper(this, "THelper", null, function () {',
  23769. ' this.GetSize = function () {',
  23770. ' var Result = 0;',
  23771. ' Result = $mod.THelper.GetSize.call(this) + 1;',
  23772. ' $mod.THelper.SetSize.call(this, 2);',
  23773. ' Result = $mod.THelper.GetSize.call(this) + 3;',
  23774. ' $mod.THelper.SetSize.call(this, 4);',
  23775. ' var $with = this.get();',
  23776. ' Result = $mod.THelper.GetSize.call(this) + 5;',
  23777. ' $mod.THelper.SetSize.call(this, 6);',
  23778. ' return Result;',
  23779. ' };',
  23780. ' this.SetSize = function (Value) {',
  23781. ' };',
  23782. '});',
  23783. 'this.w = 0;',
  23784. '']),
  23785. LinesToStr([ // $mod.$main
  23786. '$mod.w = $mod.THelper.GetSize.call({',
  23787. ' p: $mod,',
  23788. ' get: function () {',
  23789. ' return this.p.w;',
  23790. ' },',
  23791. ' set: function (v) {',
  23792. ' this.p.w = v;',
  23793. ' }',
  23794. '}) + 7;',
  23795. '$mod.THelper.SetSize.call({',
  23796. ' p: $mod,',
  23797. ' get: function () {',
  23798. ' return this.p.w;',
  23799. ' },',
  23800. ' set: function (v) {',
  23801. ' this.p.w = v;',
  23802. ' }',
  23803. '}, $mod.w + 8);',
  23804. 'var $with = $mod.w;',
  23805. '$mod.w = $mod.THelper.GetSize.call({',
  23806. ' get: function () {',
  23807. ' return $with;',
  23808. ' },',
  23809. ' set: function (v) {',
  23810. ' $with = v;',
  23811. ' }',
  23812. '}) + 9;',
  23813. '$mod.THelper.SetSize.call({',
  23814. ' get: function () {',
  23815. ' return $with;',
  23816. ' },',
  23817. ' set: function (v) {',
  23818. ' $with = v;',
  23819. ' }',
  23820. '}, $mod.w + 10);',
  23821. '']));
  23822. end;
  23823. procedure TTestModule.TestTypeHelper_Property_Array;
  23824. begin
  23825. StartProgram(false);
  23826. Add([
  23827. '{$modeswitch typehelpers}',
  23828. 'type',
  23829. ' THelper = type helper for word',
  23830. ' function GetItems(Index: byte): boolean;',
  23831. ' procedure SetItems(Index: byte; Value: boolean);',
  23832. ' property Items[Index: byte]: boolean read GetItems write SetItems;',
  23833. ' end;',
  23834. 'function THelper.GetItems(Index: byte): boolean;',
  23835. 'begin',
  23836. ' Result:=Items[1];',
  23837. ' Items[2]:=false;',
  23838. ' Result:=Self.Items[3];',
  23839. ' Self.Items[4]:=true;',
  23840. ' with Self do begin',
  23841. ' Result:=Items[5];',
  23842. ' Items[6]:=false;',
  23843. ' end;',
  23844. 'end;',
  23845. 'procedure THelper.SetItems(Index: byte; Value: boolean);',
  23846. 'begin',
  23847. 'end;',
  23848. 'var',
  23849. ' w: word;',
  23850. ' b: boolean;',
  23851. 'begin',
  23852. ' b:=w.Items[1];',
  23853. ' w.Items[2]:=b;',
  23854. ' with w do begin',
  23855. ' b:=Items[3];',
  23856. ' Items[4]:=b;',
  23857. ' end;',
  23858. '']);
  23859. ConvertProgram;
  23860. CheckSource('TestTypeHelper_Property_Array',
  23861. LinesToStr([ // statements
  23862. 'rtl.createHelper(this, "THelper", null, function () {',
  23863. ' this.GetItems = function (Index) {',
  23864. ' var Result = false;',
  23865. ' Result = $mod.THelper.GetItems.call(this, 1);',
  23866. ' $mod.THelper.SetItems.call(this, 2, false);',
  23867. ' Result = $mod.THelper.GetItems.call(this, 3);',
  23868. ' $mod.THelper.SetItems.call(this, 4, true);',
  23869. ' var $with = this.get();',
  23870. ' Result = $mod.THelper.GetItems.call(this, 5);',
  23871. ' $mod.THelper.SetItems.call(this, 6, false);',
  23872. ' return Result;',
  23873. ' };',
  23874. ' this.SetItems = function (Index, Value) {',
  23875. ' };',
  23876. '});',
  23877. 'this.w = 0;',
  23878. 'this.b = false;',
  23879. '']),
  23880. LinesToStr([ // $mod.$main
  23881. '$mod.b = $mod.THelper.GetItems.call({',
  23882. ' p: $mod,',
  23883. ' get: function () {',
  23884. ' return this.p.w;',
  23885. ' },',
  23886. ' set: function (v) {',
  23887. ' this.p.w = v;',
  23888. ' }',
  23889. '}, 1);',
  23890. '$mod.THelper.SetItems.call({',
  23891. ' p: $mod,',
  23892. ' get: function () {',
  23893. ' return this.p.w;',
  23894. ' },',
  23895. ' set: function (v) {',
  23896. ' this.p.w = v;',
  23897. ' }',
  23898. '}, 2, $mod.b);',
  23899. 'var $with = $mod.w;',
  23900. '$mod.b = $mod.THelper.GetItems.call({',
  23901. ' get: function () {',
  23902. ' return $with;',
  23903. ' },',
  23904. ' set: function (v) {',
  23905. ' $with = v;',
  23906. ' }',
  23907. '}, 3);',
  23908. '$mod.THelper.SetItems.call({',
  23909. ' get: function () {',
  23910. ' return $with;',
  23911. ' },',
  23912. ' set: function (v) {',
  23913. ' $with = v;',
  23914. ' }',
  23915. '}, 4, $mod.b);',
  23916. '']));
  23917. end;
  23918. procedure TTestModule.TestTypeHelper_ClassProperty;
  23919. begin
  23920. StartProgram(false);
  23921. Add([
  23922. '{$modeswitch typehelpers}',
  23923. 'type',
  23924. ' THelper = type helper for word',
  23925. ' class function GetSize: longint; static;',
  23926. ' class procedure SetSize(Value: longint); static;',
  23927. ' class property Size: longint read GetSize write SetSize;',
  23928. ' end;',
  23929. 'class function THelper.GetSize: longint;',
  23930. 'begin',
  23931. ' Result:=Size+1;',
  23932. ' Size:=2;',
  23933. 'end;',
  23934. 'class procedure THelper.SetSize(Value: longint);',
  23935. 'begin',
  23936. 'end;',
  23937. 'begin',
  23938. '']);
  23939. ConvertProgram;
  23940. CheckSource('TestTypeHelper_ClassProperty',
  23941. LinesToStr([ // statements
  23942. 'rtl.createHelper(this, "THelper", null, function () {',
  23943. ' this.GetSize = function () {',
  23944. ' var Result = 0;',
  23945. ' Result = $mod.THelper.GetSize() + 1;',
  23946. ' $mod.THelper.SetSize(2);',
  23947. ' return Result;',
  23948. ' };',
  23949. ' this.SetSize = function (Value) {',
  23950. ' };',
  23951. '});',
  23952. '']),
  23953. LinesToStr([ // $mod.$main
  23954. '']));
  23955. end;
  23956. procedure TTestModule.TestTypeHelper_ClassProperty_Array;
  23957. begin
  23958. StartProgram(false);
  23959. Add([
  23960. '{$modeswitch typehelpers}',
  23961. 'type',
  23962. ' THelper = type helper for word',
  23963. ' class function GetItems(Index: byte): boolean; static;',
  23964. ' class procedure SetItems(Index: byte; Value: boolean); static;',
  23965. ' class property Items[Index: byte]: boolean read GetItems write SetItems;',
  23966. ' end;',
  23967. 'class function THelper.GetItems(Index: byte): boolean;',
  23968. 'begin',
  23969. ' Result:=Items[1];',
  23970. ' Items[2]:=false;',
  23971. 'end;',
  23972. 'class procedure THelper.SetItems(Index: byte; Value: boolean);',
  23973. 'begin',
  23974. 'end;',
  23975. 'var',
  23976. ' w: word;',
  23977. ' b: boolean;',
  23978. 'begin',
  23979. ' b:=w.Items[1];',
  23980. ' w.Items[2]:=b;',
  23981. ' with w do begin',
  23982. ' b:=Items[3];',
  23983. ' Items[4]:=b;',
  23984. ' end;',
  23985. '']);
  23986. ConvertProgram;
  23987. CheckSource('TestTypeHelper_ClassProperty_Array',
  23988. LinesToStr([ // statements
  23989. 'rtl.createHelper(this, "THelper", null, function () {',
  23990. ' this.GetItems = function (Index) {',
  23991. ' var Result = false;',
  23992. ' Result = $mod.THelper.GetItems(1);',
  23993. ' $mod.THelper.SetItems(2, false);',
  23994. ' return Result;',
  23995. ' };',
  23996. ' this.SetItems = function (Index, Value) {',
  23997. ' };',
  23998. '});',
  23999. 'this.w = 0;',
  24000. 'this.b = false;',
  24001. '']),
  24002. LinesToStr([ // $mod.$main
  24003. '$mod.b = $mod.THelper.GetItems(1);',
  24004. '$mod.THelper.SetItems(2, $mod.b);',
  24005. 'var $with = $mod.w;',
  24006. '$mod.b = $mod.THelper.GetItems(3);',
  24007. '$mod.THelper.SetItems(4, $mod.b);',
  24008. '']));
  24009. end;
  24010. procedure TTestModule.TestTypeHelper_ClassMethod;
  24011. begin
  24012. StartProgram(false);
  24013. Add([
  24014. '{$modeswitch typehelpers}',
  24015. 'type',
  24016. ' THelper = type helper for word',
  24017. ' class procedure DoStatic; static;',
  24018. ' end;',
  24019. 'class procedure THelper.DoStatic;',
  24020. 'begin',
  24021. ' DoStatic;',
  24022. ' DoStatic();',
  24023. 'end;',
  24024. 'var w: word;',
  24025. 'begin',
  24026. ' w.DoStatic;',
  24027. ' w.DoStatic();',
  24028. '']);
  24029. ConvertProgram;
  24030. CheckSource('TestTypeHelper_ClassMethod',
  24031. LinesToStr([ // statements
  24032. 'rtl.createHelper(this, "THelper", null, function () {',
  24033. ' this.DoStatic = function () {',
  24034. ' $mod.THelper.DoStatic();',
  24035. ' $mod.THelper.DoStatic();',
  24036. ' };',
  24037. '});',
  24038. 'this.w = 0;',
  24039. '']),
  24040. LinesToStr([ // $mod.$main
  24041. '$mod.THelper.DoStatic();',
  24042. '$mod.THelper.DoStatic();',
  24043. '']));
  24044. end;
  24045. procedure TTestModule.TestTypeHelper_ExtClassMethodFail;
  24046. begin
  24047. StartProgram(false);
  24048. Add([
  24049. '{$modeswitch typehelpers}',
  24050. 'type',
  24051. ' THelper = type helper for word',
  24052. ' procedure Run; external name ''Run'';',
  24053. ' end;',
  24054. 'var w: word;',
  24055. 'begin',
  24056. ' w.Run;',
  24057. '']);
  24058. SetExpectedPasResolverError('Not supported: external method in type helper',nNotSupportedX);
  24059. ConvertProgram;
  24060. end;
  24061. procedure TTestModule.TestTypeHelper_Constructor;
  24062. begin
  24063. StartProgram(false);
  24064. Add([
  24065. '{$modeswitch typehelpers}',
  24066. 'type',
  24067. ' THelper = type helper for word',
  24068. ' constructor Init(e: longint);',
  24069. ' end;',
  24070. 'constructor THelper.Init(e: longint);',
  24071. 'begin',
  24072. ' Self:=e;',
  24073. ' Init(e+1);',
  24074. 'end;',
  24075. 'var w: word;',
  24076. 'begin',
  24077. ' w:=word.Init(2);',
  24078. ' w:=w.Init(3);',
  24079. ' with word do w:=Init(4);',
  24080. ' with w do w:=Init(5);',
  24081. '']);
  24082. ConvertProgram;
  24083. CheckSource('TestTypeHelper_Constructor',
  24084. LinesToStr([ // statements
  24085. 'rtl.createHelper(this, "THelper", null, function () {',
  24086. ' this.Init = function (e) {',
  24087. ' this.set(e);',
  24088. ' $mod.THelper.Init.call(this, e + 1);',
  24089. ' return this.get();',
  24090. ' };',
  24091. ' this.$new = function (fn, args) {',
  24092. ' return this[fn].apply({',
  24093. ' p: 0,',
  24094. ' get: function () {',
  24095. ' return this.p;',
  24096. ' },',
  24097. ' set: function (v) {',
  24098. ' this.p = v;',
  24099. ' }',
  24100. ' }, args);',
  24101. ' };',
  24102. '});',
  24103. 'this.w = 0;',
  24104. '']),
  24105. LinesToStr([ // $mod.$main
  24106. '$mod.w = $mod.THelper.$new("Init", [2]);',
  24107. '$mod.w = $mod.THelper.Init.call({',
  24108. ' p: $mod,',
  24109. ' get: function () {',
  24110. ' return this.p.w;',
  24111. ' },',
  24112. ' set: function (v) {',
  24113. ' this.p.w = v;',
  24114. ' }',
  24115. '}, 3);',
  24116. '$mod.w = $mod.THelper.$new("Init", [4]);',
  24117. 'var $with = $mod.w;',
  24118. '$mod.w = $mod.THelper.Init.call({',
  24119. ' get: function () {',
  24120. ' return $with;',
  24121. ' },',
  24122. ' set: function (v) {',
  24123. ' $with = v;',
  24124. ' }',
  24125. '}, 5);',
  24126. '']));
  24127. end;
  24128. procedure TTestModule.TestTypeHelper_Word;
  24129. begin
  24130. StartProgram(false);
  24131. Add([
  24132. '{$modeswitch typehelpers}',
  24133. 'type',
  24134. ' THelper = type helper for word',
  24135. ' procedure DoIt(e: byte = 123);',
  24136. ' end;',
  24137. 'procedure THelper.DoIt(e: byte);',
  24138. 'begin',
  24139. ' Self:=e;',
  24140. ' Self:=Self+1;',
  24141. ' with Self do Doit;',
  24142. 'end;',
  24143. 'begin',
  24144. ' word(3).DoIt;',
  24145. '']);
  24146. ConvertProgram;
  24147. CheckSource('TestTypeHelper_Word',
  24148. LinesToStr([ // statements
  24149. 'rtl.createHelper(this, "THelper", null, function () {',
  24150. ' this.DoIt = function (e) {',
  24151. ' this.set(e);',
  24152. ' this.set(this.get() + 1);',
  24153. ' var $with = this.get();',
  24154. ' $mod.THelper.DoIt.call(this, 123);',
  24155. ' };',
  24156. '});',
  24157. '']),
  24158. LinesToStr([ // $mod.$main
  24159. '$mod.THelper.DoIt.call({',
  24160. ' get: function () {',
  24161. ' return 3;',
  24162. ' },',
  24163. ' set: function (v) {',
  24164. ' rtl.raiseE("EPropReadOnly");',
  24165. ' }',
  24166. '}, 123);',
  24167. '']));
  24168. end;
  24169. procedure TTestModule.TestTypeHelper_Boolean;
  24170. begin
  24171. StartProgram(false);
  24172. Add([
  24173. '{$modeswitch typehelpers}',
  24174. 'type',
  24175. ' Integer = longint;',
  24176. ' THelper = type helper for boolean',
  24177. ' procedure Run(e: wordbool = true);',
  24178. ' end;',
  24179. 'procedure THelper.Run(e: wordbool);',
  24180. 'begin',
  24181. ' Self:=e;',
  24182. ' Self:=not Self;',
  24183. ' with Self do Run;',
  24184. ' if Integer(Self)=0 then ;',
  24185. 'end;',
  24186. 'begin',
  24187. ' boolean(3).Run;',
  24188. '']);
  24189. ConvertProgram;
  24190. CheckSource('TestTypeHelper_Boolean',
  24191. LinesToStr([ // statements
  24192. 'rtl.createHelper(this, "THelper", null, function () {',
  24193. ' this.Run = function (e) {',
  24194. ' this.set(e);',
  24195. ' this.set(!this.get());',
  24196. ' var $with = this.get();',
  24197. ' $mod.THelper.Run.call(this, true);',
  24198. ' if ((this.get() ? 1 : 0) === 0) ;',
  24199. ' };',
  24200. '});',
  24201. '']),
  24202. LinesToStr([ // $mod.$main
  24203. '$mod.THelper.Run.call({',
  24204. ' a: 3 != 0,',
  24205. ' get: function () {',
  24206. ' return this.a;',
  24207. ' },',
  24208. ' set: function (v) {',
  24209. ' rtl.raiseE("EPropReadOnly");',
  24210. ' }',
  24211. '}, true);',
  24212. '']));
  24213. end;
  24214. procedure TTestModule.TestTypeHelper_WordBool;
  24215. begin
  24216. StartProgram(false);
  24217. Add([
  24218. '{$modeswitch typehelpers}',
  24219. 'type',
  24220. ' Integer = longint;',
  24221. ' THelper = type helper for WordBool',
  24222. ' procedure Run(e: wordbool = true);',
  24223. ' end;',
  24224. 'procedure THelper.Run(e: wordbool);',
  24225. 'var i: integer;',
  24226. 'begin',
  24227. ' i:=Integer(Self);',
  24228. 'end;',
  24229. 'var w: wordbool;',
  24230. 'begin',
  24231. ' w.Run;',
  24232. ' wordbool(3).Run;',
  24233. '']);
  24234. ConvertProgram;
  24235. CheckSource('TestTypeHelper_WordBool',
  24236. LinesToStr([ // statements
  24237. 'rtl.createHelper(this, "THelper", null, function () {',
  24238. ' this.Run = function (e) {',
  24239. ' var i = 0;',
  24240. ' i = (this.get() ? 1 : 0);',
  24241. ' };',
  24242. '});',
  24243. 'this.w = false;',
  24244. '']),
  24245. LinesToStr([ // $mod.$main
  24246. '$mod.THelper.Run.call({',
  24247. ' p: $mod,',
  24248. ' get: function () {',
  24249. ' return this.p.w;',
  24250. ' },',
  24251. ' set: function (v) {',
  24252. ' this.p.w = v;',
  24253. ' }',
  24254. '}, true);',
  24255. '$mod.THelper.Run.call({',
  24256. ' a: 3 != 0,',
  24257. ' get: function () {',
  24258. ' return this.a;',
  24259. ' },',
  24260. ' set: function (v) {',
  24261. ' rtl.raiseE("EPropReadOnly");',
  24262. ' }',
  24263. '}, true);',
  24264. '']));
  24265. end;
  24266. procedure TTestModule.TestTypeHelper_Double;
  24267. begin
  24268. StartProgram(false);
  24269. Add([
  24270. '{$modeswitch typehelpers}',
  24271. 'type',
  24272. ' Float = type double;',
  24273. ' THelper = type helper for Float',
  24274. ' const NPI = 3.141592;',
  24275. ' function ToStr: String;',
  24276. ' end;',
  24277. 'function THelper.ToStr: String;',
  24278. 'begin',
  24279. 'end;',
  24280. 'procedure DoIt(s: string);',
  24281. 'begin',
  24282. 'end;',
  24283. 'var f: Float;',
  24284. 'begin',
  24285. ' DoIt(f.toStr);',
  24286. ' DoIt(f.toStr());',
  24287. ' (f*f).toStr;',
  24288. ' DoIt((f*f).toStr);',
  24289. '']);
  24290. ConvertProgram;
  24291. CheckSource('TestTypeHelper_Double',
  24292. LinesToStr([ // statements
  24293. 'rtl.createHelper(this, "THelper", null, function () {',
  24294. ' this.NPI = 3.141592;',
  24295. ' this.ToStr = function () {',
  24296. ' var Result = "";',
  24297. ' return Result;',
  24298. ' };',
  24299. '});',
  24300. 'this.DoIt = function (s) {',
  24301. '};',
  24302. 'this.f = 0.0;',
  24303. '']),
  24304. LinesToStr([ // $mod.$main
  24305. '$mod.DoIt($mod.THelper.ToStr.call({',
  24306. ' p: $mod,',
  24307. ' get: function () {',
  24308. ' return this.p.f;',
  24309. ' },',
  24310. ' set: function (v) {',
  24311. ' this.p.f = v;',
  24312. ' }',
  24313. '}));',
  24314. '$mod.DoIt($mod.THelper.ToStr.call({',
  24315. ' p: $mod,',
  24316. ' get: function () {',
  24317. ' return this.p.f;',
  24318. ' },',
  24319. ' set: function (v) {',
  24320. ' this.p.f = v;',
  24321. ' }',
  24322. '}));',
  24323. '$mod.THelper.ToStr.call({',
  24324. ' a: $mod.f * $mod.f,',
  24325. ' get: function () {',
  24326. ' return this.a;',
  24327. ' },',
  24328. ' set: function (v) {',
  24329. ' rtl.raiseE("EPropReadOnly");',
  24330. ' }',
  24331. '});',
  24332. '$mod.DoIt($mod.THelper.ToStr.call({',
  24333. ' a: $mod.f * $mod.f,',
  24334. ' get: function () {',
  24335. ' return this.a;',
  24336. ' },',
  24337. ' set: function (v) {',
  24338. ' rtl.raiseE("EPropReadOnly");',
  24339. ' }',
  24340. '}));',
  24341. '']));
  24342. end;
  24343. procedure TTestModule.TestTypeHelper_NativeInt;
  24344. begin
  24345. StartProgram(false);
  24346. Add([
  24347. '{$modeswitch typehelpers}',
  24348. 'type',
  24349. ' MaxInt = type nativeint;',
  24350. ' THelperI = type helper for MaxInt',
  24351. ' function ToStr: String;',
  24352. ' end;',
  24353. ' MaxUInt = type nativeuint;',
  24354. ' THelperU = type helper for MaxUInt',
  24355. ' function ToStr: String;',
  24356. ' end;',
  24357. 'function THelperI.ToStr: String;',
  24358. 'begin',
  24359. ' Result:=str(Self);',
  24360. 'end;',
  24361. 'function THelperU.ToStr: String;',
  24362. 'begin',
  24363. ' Result:=str(Self);',
  24364. 'end;',
  24365. 'procedure DoIt(s: string);',
  24366. 'begin',
  24367. 'end;',
  24368. 'var i: MaxInt;',
  24369. 'begin',
  24370. ' DoIt(i.toStr);',
  24371. ' DoIt(i.toStr());',
  24372. ' (i*i).toStr;',
  24373. ' DoIt((i*i).toStr);',
  24374. '']);
  24375. ConvertProgram;
  24376. CheckSource('TestTypeHelper_NativeInt',
  24377. LinesToStr([ // statements
  24378. 'rtl.createHelper(this, "THelperI", null, function () {',
  24379. ' this.ToStr = function () {',
  24380. ' var Result = "";',
  24381. ' Result = "" + this.get();',
  24382. ' return Result;',
  24383. ' };',
  24384. '});',
  24385. 'rtl.createHelper(this, "THelperU", null, function () {',
  24386. ' this.ToStr = function () {',
  24387. ' var Result = "";',
  24388. ' Result = "" + this.get();',
  24389. ' return Result;',
  24390. ' };',
  24391. '});',
  24392. 'this.DoIt = function (s) {',
  24393. '};',
  24394. 'this.i = 0;',
  24395. '']),
  24396. LinesToStr([ // $mod.$main
  24397. '$mod.DoIt($mod.THelperI.ToStr.call({',
  24398. ' p: $mod,',
  24399. ' get: function () {',
  24400. ' return this.p.i;',
  24401. ' },',
  24402. ' set: function (v) {',
  24403. ' this.p.i = v;',
  24404. ' }',
  24405. '}));',
  24406. '$mod.DoIt($mod.THelperI.ToStr.call({',
  24407. ' p: $mod,',
  24408. ' get: function () {',
  24409. ' return this.p.i;',
  24410. ' },',
  24411. ' set: function (v) {',
  24412. ' this.p.i = v;',
  24413. ' }',
  24414. '}));',
  24415. '$mod.THelperI.ToStr.call({',
  24416. ' a: $mod.i * $mod.i,',
  24417. ' get: function () {',
  24418. ' return this.a;',
  24419. ' },',
  24420. ' set: function (v) {',
  24421. ' rtl.raiseE("EPropReadOnly");',
  24422. ' }',
  24423. '});',
  24424. '$mod.DoIt($mod.THelperI.ToStr.call({',
  24425. ' a: $mod.i * $mod.i,',
  24426. ' get: function () {',
  24427. ' return this.a;',
  24428. ' },',
  24429. ' set: function (v) {',
  24430. ' rtl.raiseE("EPropReadOnly");',
  24431. ' }',
  24432. '}));',
  24433. '']));
  24434. end;
  24435. procedure TTestModule.TestTypeHelper_StringChar;
  24436. begin
  24437. StartProgram(false);
  24438. Add([
  24439. '{$modeswitch typehelpers}',
  24440. 'type',
  24441. ' TStringHelper = type helper for string',
  24442. ' procedure DoIt(e: byte = 123);',
  24443. ' end;',
  24444. ' TCharHelper = type helper for char',
  24445. ' procedure Fly;',
  24446. ' end;',
  24447. 'procedure TStringHelper.DoIt(e: byte);',
  24448. 'begin',
  24449. ' Self[1]:=''c'';',
  24450. ' Self[2]:=Self[3];',
  24451. 'end;',
  24452. 'procedure TCharHelper.Fly;',
  24453. 'begin',
  24454. ' Self:=''c'';',
  24455. 'end;',
  24456. 'begin',
  24457. ' ''abc''.DoIt;',
  24458. ' ''xyz''.DoIt();',
  24459. ' ''c''.Fly();',
  24460. '']);
  24461. ConvertProgram;
  24462. CheckSource('TestTypeHelper_StringChar',
  24463. LinesToStr([ // statements
  24464. 'rtl.createHelper(this, "TStringHelper", null, function () {',
  24465. ' this.DoIt = function (e) {',
  24466. ' this.set(rtl.setCharAt(this.get(), 0, "c"));',
  24467. ' this.set(rtl.setCharAt(this.get(), 1, this.get().charAt(2)));',
  24468. ' };',
  24469. '});',
  24470. 'rtl.createHelper(this, "TCharHelper", null, function () {',
  24471. ' this.Fly = function () {',
  24472. ' this.set("c");',
  24473. ' };',
  24474. '});',
  24475. '']),
  24476. LinesToStr([ // $mod.$main
  24477. '$mod.TStringHelper.DoIt.call({',
  24478. ' get: function () {',
  24479. ' return "abc";',
  24480. ' },',
  24481. ' set: function (v) {',
  24482. ' rtl.raiseE("EPropReadOnly");',
  24483. ' }',
  24484. '}, 123);',
  24485. '$mod.TStringHelper.DoIt.call({',
  24486. ' get: function () {',
  24487. ' return "xyz";',
  24488. ' },',
  24489. ' set: function (v) {',
  24490. ' rtl.raiseE("EPropReadOnly");',
  24491. ' }',
  24492. '}, 123);',
  24493. '$mod.TCharHelper.Fly.call({',
  24494. ' get: function () {',
  24495. ' return "c";',
  24496. ' },',
  24497. ' set: function (v) {',
  24498. ' rtl.raiseE("EPropReadOnly");',
  24499. ' }',
  24500. '});',
  24501. '']));
  24502. end;
  24503. procedure TTestModule.TestTypeHelper_JSValue;
  24504. begin
  24505. StartProgram(false);
  24506. Add([
  24507. '{$modeswitch typehelpers}',
  24508. 'type',
  24509. ' TExtValue = type jsvalue;',
  24510. ' THelper = type helper for TExtValue',
  24511. ' function ToStr: String;',
  24512. ' end;',
  24513. 'function THelper.ToStr: String;',
  24514. 'begin',
  24515. 'end;',
  24516. 'var',
  24517. ' s: string;',
  24518. ' v: TExtValue;',
  24519. 'begin',
  24520. ' s:=v.toStr;',
  24521. ' s:=v.toStr();',
  24522. ' TExtValue(s).toStr;',
  24523. '']);
  24524. ConvertProgram;
  24525. CheckSource('TestTypeHelper_JSValue',
  24526. LinesToStr([ // statements
  24527. 'rtl.createHelper(this, "THelper", null, function () {',
  24528. ' this.ToStr = function () {',
  24529. ' var Result = "";',
  24530. ' return Result;',
  24531. ' };',
  24532. '});',
  24533. 'this.s = "";',
  24534. 'this.v = undefined;',
  24535. '']),
  24536. LinesToStr([ // $mod.$main
  24537. '$mod.s = $mod.THelper.ToStr.call({',
  24538. ' p: $mod,',
  24539. ' get: function () {',
  24540. ' return this.p.v;',
  24541. ' },',
  24542. ' set: function (v) {',
  24543. ' this.p.v = v;',
  24544. ' }',
  24545. '});',
  24546. '$mod.s = $mod.THelper.ToStr.call({',
  24547. ' p: $mod,',
  24548. ' get: function () {',
  24549. ' return this.p.v;',
  24550. ' },',
  24551. ' set: function (v) {',
  24552. ' this.p.v = v;',
  24553. ' }',
  24554. '});',
  24555. '$mod.THelper.ToStr.call({',
  24556. ' p: $mod,',
  24557. ' get: function () {',
  24558. ' return this.p.s;',
  24559. ' },',
  24560. ' set: function (v) {',
  24561. ' rtl.raiseE("EPropReadOnly");',
  24562. ' }',
  24563. '});',
  24564. '']));
  24565. end;
  24566. procedure TTestModule.TestTypeHelper_Array;
  24567. begin
  24568. StartProgram(false);
  24569. Add([
  24570. '{$modeswitch typehelpers}',
  24571. 'type',
  24572. ' TArrOfBool = array of boolean;',
  24573. ' TArrOfJS = array of jsvalue;',
  24574. ' THelper = type helper for TArrOfBool',
  24575. ' procedure DoIt(e: byte = 123);',
  24576. ' end;',
  24577. 'procedure THelper.DoIt(e: byte);',
  24578. 'begin',
  24579. ' Self[1]:=true;',
  24580. ' Self[2]:=not Self[3];',
  24581. ' SetLength(Self,4);',
  24582. 'end;',
  24583. 'var',
  24584. ' b: TArrOfBool;',
  24585. ' j: TArrOfJS;',
  24586. 'begin',
  24587. ' b.DoIt;',
  24588. ' TArrOfBool(j).DoIt();',
  24589. '']);
  24590. ConvertProgram;
  24591. CheckSource('TestTypeHelper_Array',
  24592. LinesToStr([ // statements
  24593. 'rtl.createHelper(this, "THelper", null, function () {',
  24594. ' this.DoIt = function (e) {',
  24595. ' this.get()[1] = true;',
  24596. ' this.get()[2] = !this.get()[3];',
  24597. ' this.set(rtl.arraySetLength(this.get(), false, 4));',
  24598. ' };',
  24599. '});',
  24600. 'this.b = [];',
  24601. 'this.j = [];',
  24602. '']),
  24603. LinesToStr([ // $mod.$main
  24604. '$mod.THelper.DoIt.call({',
  24605. ' p: $mod,',
  24606. ' get: function () {',
  24607. ' return this.p.b;',
  24608. ' },',
  24609. ' set: function (v) {',
  24610. ' this.p.b = v;',
  24611. ' }',
  24612. '}, 123);',
  24613. '$mod.THelper.DoIt.call({',
  24614. ' p: $mod,',
  24615. ' get: function () {',
  24616. ' return this.p.j;',
  24617. ' },',
  24618. ' set: function (v) {',
  24619. ' this.p.j = v;',
  24620. ' }',
  24621. '}, 123);',
  24622. '']));
  24623. end;
  24624. procedure TTestModule.TestTypeHelper_EnumType;
  24625. begin
  24626. StartProgram(false);
  24627. Add([
  24628. '{$modeswitch typehelpers}',
  24629. 'type',
  24630. ' TEnum = (red,blue);',
  24631. ' THelper = type helper for TEnum',
  24632. ' procedure DoIt(e: byte = 123);',
  24633. ' class procedure Swing(w: word); static;',
  24634. ' end;',
  24635. 'procedure THelper.DoIt(e: byte);',
  24636. 'begin',
  24637. ' Self:=red;',
  24638. ' Self:=succ(Self);',
  24639. ' with Self do Doit;',
  24640. 'end;',
  24641. 'class procedure THelper.Swing(w: word);',
  24642. 'begin',
  24643. 'end;',
  24644. 'var e: TEnum;',
  24645. 'begin',
  24646. ' e.DoIt;',
  24647. ' red.DoIt;',
  24648. ' TEnum.blue.DoIt;',
  24649. ' TEnum(1).DoIt;',
  24650. ' TEnum.Swing(3);',
  24651. '']);
  24652. ConvertProgram;
  24653. CheckSource('TestTypeHelper_EnumType',
  24654. LinesToStr([ // statements
  24655. 'this.TEnum = {',
  24656. ' "0": "red",',
  24657. ' red: 0,',
  24658. ' "1": "blue",',
  24659. ' blue: 1',
  24660. '};',
  24661. 'rtl.createHelper(this, "THelper", null, function () {',
  24662. ' this.DoIt = function (e) {',
  24663. ' this.set($mod.TEnum.red);',
  24664. ' this.set(this.get() + 1);',
  24665. ' var $with = this.get();',
  24666. ' $mod.THelper.DoIt.call(this, 123);',
  24667. ' };',
  24668. ' this.Swing = function (w) {',
  24669. ' };',
  24670. '});',
  24671. 'this.e = 0;',
  24672. '']),
  24673. LinesToStr([ // $mod.$main
  24674. '$mod.THelper.DoIt.call({',
  24675. ' p: $mod,',
  24676. ' get: function () {',
  24677. ' return this.p.e;',
  24678. ' },',
  24679. ' set: function (v) {',
  24680. ' this.p.e = v;',
  24681. ' }',
  24682. '}, 123);',
  24683. '$mod.THelper.DoIt.call({',
  24684. ' p: $mod.TEnum,',
  24685. ' get: function () {',
  24686. ' return this.p.red;',
  24687. ' },',
  24688. ' set: function (v) {',
  24689. ' rtl.raiseE("EPropReadOnly");',
  24690. ' }',
  24691. '}, 123);',
  24692. '$mod.THelper.DoIt.call({',
  24693. ' p: $mod.TEnum,',
  24694. ' get: function () {',
  24695. ' return this.p.blue;',
  24696. ' },',
  24697. ' set: function (v) {',
  24698. ' rtl.raiseE("EPropReadOnly");',
  24699. ' }',
  24700. '}, 123);',
  24701. '$mod.THelper.DoIt.call({',
  24702. ' get: function () {',
  24703. ' return 1;',
  24704. ' },',
  24705. ' set: function (v) {',
  24706. ' rtl.raiseE("EPropReadOnly");',
  24707. ' }',
  24708. '}, 123);',
  24709. '$mod.THelper.Swing(3);',
  24710. '']));
  24711. end;
  24712. procedure TTestModule.TestTypeHelper_SetType;
  24713. begin
  24714. StartProgram(false);
  24715. Add([
  24716. '{$modeswitch typehelpers}',
  24717. 'type',
  24718. ' TEnum = (red,blue);',
  24719. ' TSetOfEnum = set of TEnum;',
  24720. ' THelper = type helper for TSetOfEnum',
  24721. ' procedure DoIt(e: byte = 123);',
  24722. ' constructor Init(e: TEnum);',
  24723. ' constructor InitEmpty;',
  24724. ' end;',
  24725. 'procedure THelper.DoIt(e: byte);',
  24726. 'begin',
  24727. ' Self:=[];',
  24728. ' Self:=[red];',
  24729. ' Include(Self,blue);',
  24730. 'end;',
  24731. 'constructor THelper.Init(e: TEnum);',
  24732. 'begin',
  24733. ' Self:=[];',
  24734. ' Self:=[e];',
  24735. ' Include(Self,blue);',
  24736. 'end;',
  24737. 'constructor THelper.InitEmpty;',
  24738. 'begin',
  24739. 'end;',
  24740. 'var s: TSetOfEnum;',
  24741. 'begin',
  24742. ' s.DoIt;',
  24743. //' [red].DoIt;',
  24744. //' with s do DoIt;',
  24745. //' with [red,blue] do DoIt;',
  24746. ' s:=TSetOfEnum.Init(blue);',
  24747. ' s:=s.Init(blue);',
  24748. '']);
  24749. ConvertProgram;
  24750. CheckSource('TestTypeHelper_SetType',
  24751. LinesToStr([ // statements
  24752. 'this.TEnum = {',
  24753. ' "0": "red",',
  24754. ' red: 0,',
  24755. ' "1": "blue",',
  24756. ' blue: 1',
  24757. '};',
  24758. 'rtl.createHelper(this, "THelper", null, function () {',
  24759. ' this.DoIt = function (e) {',
  24760. ' this.set({});',
  24761. ' this.set(rtl.createSet($mod.TEnum.red));',
  24762. ' this.set(rtl.includeSet(this.get(), $mod.TEnum.blue));',
  24763. ' };',
  24764. ' this.Init = function (e) {',
  24765. ' this.set({});',
  24766. ' this.set(rtl.createSet(e));',
  24767. ' this.set(rtl.includeSet(this.get(), $mod.TEnum.blue));',
  24768. ' return this.get();',
  24769. ' };',
  24770. ' this.InitEmpty = function () {',
  24771. ' return this.get();',
  24772. ' };',
  24773. ' this.$new = function (fn, args) {',
  24774. ' return this[fn].apply({',
  24775. ' p: {},',
  24776. ' get: function () {',
  24777. ' return this.p;',
  24778. ' },',
  24779. ' set: function (v) {',
  24780. ' this.p = v;',
  24781. ' }',
  24782. ' }, args);',
  24783. ' };',
  24784. '});',
  24785. 'this.s = {};',
  24786. '']),
  24787. LinesToStr([ // $mod.$main
  24788. '$mod.THelper.DoIt.call({',
  24789. ' p: $mod,',
  24790. ' get: function () {',
  24791. ' return this.p.s;',
  24792. ' },',
  24793. ' set: function (v) {',
  24794. ' this.p.s = v;',
  24795. ' }',
  24796. '}, 123);',
  24797. '$mod.s = rtl.refSet($mod.THelper.$new("Init", [$mod.TEnum.blue]));',
  24798. '$mod.s = rtl.refSet($mod.THelper.Init.call({',
  24799. ' p: $mod,',
  24800. ' get: function () {',
  24801. ' return this.p.s;',
  24802. ' },',
  24803. ' set: function (v) {',
  24804. ' this.p.s = v;',
  24805. ' }',
  24806. '}, $mod.TEnum.blue));',
  24807. '']));
  24808. end;
  24809. procedure TTestModule.TestTypeHelper_InterfaceType;
  24810. begin
  24811. StartProgram(false);
  24812. Add([
  24813. '{$interfaces com}',
  24814. '{$modeswitch typehelpers}',
  24815. 'type',
  24816. ' IUnknown = interface',
  24817. ' function _AddRef: longint;',
  24818. ' function _Release: longint;',
  24819. ' end;',
  24820. ' TObject = class(IUnknown)',
  24821. ' function _AddRef: longint; virtual; abstract;',
  24822. ' function _Release: longint; virtual; abstract;',
  24823. ' end;',
  24824. ' THelper = type helper for IUnknown',
  24825. ' procedure Fly(e: byte = 123);',
  24826. ' class procedure Run; static;',
  24827. ' end;',
  24828. 'var',
  24829. ' i: IUnknown;',
  24830. ' o: TObject;',
  24831. 'procedure THelper.Fly(e: byte);',
  24832. 'begin',
  24833. ' i:=Self;',
  24834. ' o:=Self as TObject;',
  24835. ' Self:=nil;',
  24836. ' Self:=i;',
  24837. ' Self:=o;',
  24838. ' with Self do begin',
  24839. ' Fly;',
  24840. ' Fly();',
  24841. ' end;',
  24842. 'end;',
  24843. 'class procedure THelper.Run;',
  24844. 'var l: IUnknown;',
  24845. 'begin',
  24846. ' l.Fly;',
  24847. ' l.Fly();',
  24848. 'end;',
  24849. 'begin',
  24850. ' i.Fly;',
  24851. ' i.Fly();',
  24852. ' i.Run;',
  24853. ' i.Run();',
  24854. ' IUnknown.Run;',
  24855. ' IUnknown.Run();',
  24856. '']);
  24857. ConvertProgram;
  24858. CheckSource('TestTypeHelper_InterfaceType',
  24859. LinesToStr([ // statements
  24860. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  24861. 'rtl.createClass(this, "TObject", null, function () {',
  24862. ' this.$init = function () {',
  24863. ' };',
  24864. ' this.$final = function () {',
  24865. ' };',
  24866. ' rtl.addIntf(this, $mod.IUnknown);',
  24867. '});',
  24868. 'rtl.createHelper(this, "THelper", null, function () {',
  24869. ' this.Fly = function (e) {',
  24870. ' var $ir = rtl.createIntfRefs();',
  24871. ' try {',
  24872. ' rtl.setIntfP($mod, "i", this.get());',
  24873. ' $mod.o = rtl.intfAsClass(this.get(), $mod.TObject);',
  24874. ' this.set(null);',
  24875. ' this.set($mod.i);',
  24876. ' this.set($ir.ref(1, rtl.queryIntfT($mod.o, $mod.IUnknown)));',
  24877. ' var $with = this.get();',
  24878. ' $mod.THelper.Fly.call(this, 123);',
  24879. ' $mod.THelper.Fly.call(this, 123);',
  24880. ' } finally {',
  24881. ' $ir.free();',
  24882. ' };',
  24883. ' };',
  24884. ' this.Run = function () {',
  24885. ' var l = null;',
  24886. ' try {',
  24887. ' $mod.THelper.Fly.call({',
  24888. ' get: function () {',
  24889. ' return l;',
  24890. ' },',
  24891. ' set: function (v) {',
  24892. ' l = rtl.setIntfL(l, v);',
  24893. ' }',
  24894. ' }, 123);',
  24895. ' $mod.THelper.Fly.call({',
  24896. ' get: function () {',
  24897. ' return l;',
  24898. ' },',
  24899. ' set: function (v) {',
  24900. ' l = rtl.setIntfL(l, v);',
  24901. ' }',
  24902. ' }, 123);',
  24903. ' } finally {',
  24904. ' rtl._Release(l);',
  24905. ' };',
  24906. ' };',
  24907. '});',
  24908. 'this.i = null;',
  24909. 'this.o = null;',
  24910. '']),
  24911. LinesToStr([ // $mod.$main
  24912. '$mod.THelper.Fly.call({',
  24913. ' p: $mod,',
  24914. ' get: function () {',
  24915. ' return this.p.i;',
  24916. ' },',
  24917. ' set: function (v) {',
  24918. ' rtl.setIntfP(this.p, "i", v);',
  24919. ' }',
  24920. '}, 123);',
  24921. '$mod.THelper.Fly.call({',
  24922. ' p: $mod,',
  24923. ' get: function () {',
  24924. ' return this.p.i;',
  24925. ' },',
  24926. ' set: function (v) {',
  24927. ' rtl.setIntfP(this.p, "i", v);',
  24928. ' }',
  24929. '}, 123);',
  24930. '$mod.THelper.Run();',
  24931. '$mod.THelper.Run();',
  24932. '$mod.THelper.Run();',
  24933. '$mod.THelper.Run();',
  24934. '']));
  24935. end;
  24936. procedure TTestModule.TestTypeHelper_NestedSelf;
  24937. begin
  24938. StartProgram(false);
  24939. Add([
  24940. '{$modeswitch typehelpers}',
  24941. 'type',
  24942. ' THelper = type helper for string',
  24943. ' procedure Run(Value: string);',
  24944. ' end;',
  24945. 'procedure THelper.Run(Value: string);',
  24946. ' function Sub(i: nativeint): boolean;',
  24947. ' begin',
  24948. ' Result:=Self[i+1]=Value[i];',
  24949. ' end;',
  24950. 'begin',
  24951. ' if Self[3]=Value[4] then ;',
  24952. 'end;',
  24953. 'begin',
  24954. '']);
  24955. ConvertProgram;
  24956. CheckSource('TestTypeHelper_NestedSelf',
  24957. LinesToStr([ // statements
  24958. 'rtl.createHelper(this, "THelper", null, function () {',
  24959. ' this.Run = function (Value) {',
  24960. ' var $Self = this;',
  24961. ' function Sub(i) {',
  24962. ' var Result = false;',
  24963. ' Result = $Self.get().charAt((i + 1) - 1) === Value.charAt(i - 1);',
  24964. ' return Result;',
  24965. ' };',
  24966. ' if ($Self.get().charAt(2) === Value.charAt(3)) ;',
  24967. ' };',
  24968. '});',
  24969. '']),
  24970. LinesToStr([ // $mod.$main
  24971. '']));
  24972. end;
  24973. procedure TTestModule.TestProcType;
  24974. begin
  24975. StartProgram(false);
  24976. Add([
  24977. 'type',
  24978. ' TProcInt = procedure(vI: longint = 1);',
  24979. 'procedure DoIt(vJ: longint);',
  24980. 'begin end;',
  24981. 'var',
  24982. ' b: boolean;',
  24983. ' vP, vQ: tprocint;',
  24984. 'begin',
  24985. ' vp:=nil;',
  24986. ' vp:=vp;',
  24987. ' vp:=@doit;',
  24988. ' vp;',
  24989. ' vp();',
  24990. ' vp(2);',
  24991. ' b:=vp=nil;',
  24992. ' b:=nil=vp;',
  24993. ' b:=vp=vq;',
  24994. ' b:=vp=@doit;',
  24995. ' b:=@doit=vp;',
  24996. ' b:=vp<>nil;',
  24997. ' b:=nil<>vp;',
  24998. ' b:=vp<>vq;',
  24999. ' b:=vp<>@doit;',
  25000. ' b:=@doit<>vp;',
  25001. ' b:=Assigned(vp);',
  25002. ' if Assigned(vp) then ;']);
  25003. ConvertProgram;
  25004. CheckSource('TestProcType',
  25005. LinesToStr([ // statements
  25006. 'this.DoIt = function(vJ) {',
  25007. '};',
  25008. 'this.b = false;',
  25009. 'this.vP = null;',
  25010. 'this.vQ = null;'
  25011. ]),
  25012. LinesToStr([ // $mod.$main
  25013. '$mod.vP = null;',
  25014. '$mod.vP = $mod.vP;',
  25015. '$mod.vP = $mod.DoIt;',
  25016. '$mod.vP(1);',
  25017. '$mod.vP(1);',
  25018. '$mod.vP(2);',
  25019. '$mod.b = $mod.vP === null;',
  25020. '$mod.b = null === $mod.vP;',
  25021. '$mod.b = rtl.eqCallback($mod.vP,$mod.vQ);',
  25022. '$mod.b = rtl.eqCallback($mod.vP, $mod.DoIt);',
  25023. '$mod.b = rtl.eqCallback($mod.DoIt, $mod.vP);',
  25024. '$mod.b = $mod.vP !== null;',
  25025. '$mod.b = null !== $mod.vP;',
  25026. '$mod.b = !rtl.eqCallback($mod.vP,$mod.vQ);',
  25027. '$mod.b = !rtl.eqCallback($mod.vP, $mod.DoIt);',
  25028. '$mod.b = !rtl.eqCallback($mod.DoIt, $mod.vP);',
  25029. '$mod.b = $mod.vP != null;',
  25030. 'if ($mod.vP != null) ;',
  25031. '']));
  25032. end;
  25033. procedure TTestModule.TestProcType_Arg;
  25034. begin
  25035. StartProgram(false);
  25036. Add([
  25037. 'type',
  25038. ' TProcInt = procedure(vI: longint = 1);',
  25039. 'procedure DoIt(vJ: longint); begin end;',
  25040. 'procedure DoSome(vP, vQ: TProcInt);',
  25041. 'var',
  25042. ' b: boolean;',
  25043. 'begin',
  25044. ' vp:=nil;',
  25045. ' vp:=vp;',
  25046. ' vp:=@doit;',
  25047. ' vp;',
  25048. ' vp();',
  25049. ' vp(2);',
  25050. ' b:=vp=nil;',
  25051. ' b:=nil=vp;',
  25052. ' b:=vp=vq;',
  25053. ' b:=vp=@doit;',
  25054. ' b:=@doit=vp;',
  25055. ' b:=vp<>nil;',
  25056. ' b:=nil<>vp;',
  25057. ' b:=vp<>vq;',
  25058. ' b:=vp<>@doit;',
  25059. ' b:=@doit<>vp;',
  25060. ' b:=Assigned(vp);',
  25061. ' if Assigned(vp) then ;',
  25062. 'end;',
  25063. 'begin',
  25064. ' DoSome(@DoIt,nil);']);
  25065. ConvertProgram;
  25066. CheckSource('TestProcType_Arg',
  25067. LinesToStr([ // statements
  25068. 'this.DoIt = function(vJ) {',
  25069. '};',
  25070. 'this.DoSome = function(vP, vQ) {',
  25071. ' var b = false;',
  25072. ' vP = null;',
  25073. ' vP = vP;',
  25074. ' vP = $mod.DoIt;',
  25075. ' vP(1);',
  25076. ' vP(1);',
  25077. ' vP(2);',
  25078. ' b = vP === null;',
  25079. ' b = null === vP;',
  25080. ' b = rtl.eqCallback(vP,vQ);',
  25081. ' b = rtl.eqCallback(vP, $mod.DoIt);',
  25082. ' b = rtl.eqCallback($mod.DoIt, vP);',
  25083. ' b = vP !== null;',
  25084. ' b = null !== vP;',
  25085. ' b = !rtl.eqCallback(vP, vQ);',
  25086. ' b = !rtl.eqCallback(vP, $mod.DoIt);',
  25087. ' b = !rtl.eqCallback($mod.DoIt, vP);',
  25088. ' b = vP != null;',
  25089. ' if (vP != null) ;',
  25090. '};',
  25091. '']),
  25092. LinesToStr([ // $mod.$main
  25093. '$mod.DoSome($mod.DoIt,null);',
  25094. '']));
  25095. end;
  25096. procedure TTestModule.TestProcType_FunctionFPC;
  25097. begin
  25098. StartProgram(false);
  25099. Add('type');
  25100. Add(' TFuncInt = function(vA: longint = 1): longint;');
  25101. Add('function DoIt(vI: longint): longint;');
  25102. Add('begin end;');
  25103. Add('var');
  25104. Add(' b: boolean;');
  25105. Add(' vP, vQ: tfuncint;');
  25106. Add('begin');
  25107. Add(' vp:=nil;');
  25108. Add(' vp:=vp;');
  25109. Add(' vp:=@doit;'); // ok in fpc and delphi
  25110. //Add(' vp:=doit;'); // illegal in fpc, ok in delphi
  25111. Add(' vp;'); // ok in fpc and delphi
  25112. Add(' vp();');
  25113. Add(' vp(2);');
  25114. Add(' b:=vp=nil;'); // ok in fpc, illegal in delphi
  25115. Add(' b:=nil=vp;'); // ok in fpc, illegal in delphi
  25116. Add(' b:=vp=vq;'); // in fpc compare proctypes, in delphi compare results
  25117. Add(' b:=vp=@doit;'); // ok in fpc, illegal in delphi
  25118. Add(' b:=@doit=vp;'); // ok in fpc, illegal in delphi
  25119. //Add(' b:=vp=3;'); // illegal in fpc, ok in delphi
  25120. Add(' b:=4=vp;'); // illegal in fpc, ok in delphi
  25121. Add(' b:=vp<>nil;'); // ok in fpc, illegal in delphi
  25122. Add(' b:=nil<>vp;'); // ok in fpc, illegal in delphi
  25123. Add(' b:=vp<>vq;'); // in fpc compare proctypes, in delphi compare results
  25124. Add(' b:=vp<>@doit;'); // ok in fpc, illegal in delphi
  25125. Add(' b:=@doit<>vp;'); // ok in fpc, illegal in delphi
  25126. //Add(' b:=vp<>5;'); // illegal in fpc, ok in delphi
  25127. Add(' b:=6<>vp;'); // illegal in fpc, ok in delphi
  25128. Add(' b:=Assigned(vp);');
  25129. //Add(' doit(vp);'); // illegal in fpc, ok in delphi
  25130. Add(' doit(vp());'); // ok in fpc and delphi
  25131. Add(' doit(vp(2));'); // ok in fpc and delphi
  25132. ConvertProgram;
  25133. CheckSource('TestProcType_FunctionFPC',
  25134. LinesToStr([ // statements
  25135. 'this.DoIt = function(vI) {',
  25136. ' var Result = 0;',
  25137. ' return Result;',
  25138. '};',
  25139. 'this.b = false;',
  25140. 'this.vP = null;',
  25141. 'this.vQ = null;'
  25142. ]),
  25143. LinesToStr([ // $mod.$main
  25144. '$mod.vP = null;',
  25145. '$mod.vP = $mod.vP;',
  25146. '$mod.vP = $mod.DoIt;',
  25147. '$mod.vP(1);',
  25148. '$mod.vP(1);',
  25149. '$mod.vP(2);',
  25150. '$mod.b = $mod.vP === null;',
  25151. '$mod.b = null === $mod.vP;',
  25152. '$mod.b = rtl.eqCallback($mod.vP,$mod.vQ);',
  25153. '$mod.b = rtl.eqCallback($mod.vP, $mod.DoIt);',
  25154. '$mod.b = rtl.eqCallback($mod.DoIt, $mod.vP);',
  25155. '$mod.b = 4 === $mod.vP(1);',
  25156. '$mod.b = $mod.vP !== null;',
  25157. '$mod.b = null !== $mod.vP;',
  25158. '$mod.b = !rtl.eqCallback($mod.vP,$mod.vQ);',
  25159. '$mod.b = !rtl.eqCallback($mod.vP, $mod.DoIt);',
  25160. '$mod.b = !rtl.eqCallback($mod.DoIt, $mod.vP);',
  25161. '$mod.b = 6 !== $mod.vP(1);',
  25162. '$mod.b = $mod.vP != null;',
  25163. '$mod.DoIt($mod.vP(1));',
  25164. '$mod.DoIt($mod.vP(2));',
  25165. '']));
  25166. end;
  25167. procedure TTestModule.TestProcType_FunctionDelphi;
  25168. begin
  25169. StartProgram(false);
  25170. Add('{$mode Delphi}');
  25171. Add('type');
  25172. Add(' TFuncInt = function(vA: longint = 1): longint;');
  25173. Add('function DoIt(vI: longint): longint;');
  25174. Add('begin end;');
  25175. Add('var');
  25176. Add(' b: boolean;');
  25177. Add(' vP, vQ: tfuncint;');
  25178. Add('begin');
  25179. Add(' vp:=nil;');
  25180. Add(' vp:=vp;');
  25181. Add(' vp:=@doit;'); // ok in fpc and delphi
  25182. Add(' vp:=doit;'); // illegal in fpc, ok in delphi
  25183. Add(' vp;'); // ok in fpc and delphi
  25184. Add(' vp();');
  25185. Add(' vp(2);');
  25186. //Add(' b:=vp=nil;'); // ok in fpc, illegal in delphi
  25187. //Add(' b:=nil=vp;'); // ok in fpc, illegal in delphi
  25188. Add(' b:=vp=vq;'); // in fpc compare proctypes, in delphi compare results
  25189. //Add(' b:=vp=@doit;'); // ok in fpc, illegal in delphi
  25190. //Add(' b:=@doit=vp;'); // ok in fpc, illegal in delphi
  25191. Add(' b:=vp=3;'); // illegal in fpc, ok in delphi
  25192. Add(' b:=4=vp;'); // illegal in fpc, ok in delphi
  25193. //Add(' b:=vp<>nil;'); // ok in fpc, illegal in delphi
  25194. //Add(' b:=nil<>vp;'); // ok in fpc, illegal in delphi
  25195. Add(' b:=vp<>vq;'); // in fpc compare proctypes, in delphi compare results
  25196. //Add(' b:=vp<>@doit;'); // ok in fpc, illegal in delphi
  25197. //Add(' b:=@doit<>vp;'); // ok in fpc, illegal in delphi
  25198. Add(' b:=vp<>5;'); // illegal in fpc, ok in delphi
  25199. Add(' b:=6<>vp;'); // illegal in fpc, ok in delphi
  25200. Add(' b:=Assigned(vp);');
  25201. Add(' doit(vp);'); // illegal in fpc, ok in delphi
  25202. Add(' doit(vp());'); // ok in fpc and delphi
  25203. Add(' doit(vp(2));'); // ok in fpc and delphi *)
  25204. ConvertProgram;
  25205. CheckSource('TestProcType_FunctionDelphi',
  25206. LinesToStr([ // statements
  25207. 'this.DoIt = function(vI) {',
  25208. ' var Result = 0;',
  25209. ' return Result;',
  25210. '};',
  25211. 'this.b = false;',
  25212. 'this.vP = null;',
  25213. 'this.vQ = null;'
  25214. ]),
  25215. LinesToStr([ // $mod.$main
  25216. '$mod.vP = null;',
  25217. '$mod.vP = $mod.vP;',
  25218. '$mod.vP = $mod.DoIt;',
  25219. '$mod.vP = $mod.DoIt;',
  25220. '$mod.vP(1);',
  25221. '$mod.vP(1);',
  25222. '$mod.vP(2);',
  25223. '$mod.b = $mod.vP(1) === $mod.vQ(1);',
  25224. '$mod.b = $mod.vP(1) === 3;',
  25225. '$mod.b = 4 === $mod.vP(1);',
  25226. '$mod.b = $mod.vP(1) !== $mod.vQ(1);',
  25227. '$mod.b = $mod.vP(1) !== 5;',
  25228. '$mod.b = 6 !== $mod.vP(1);',
  25229. '$mod.b = $mod.vP != null;',
  25230. '$mod.DoIt($mod.vP(1));',
  25231. '$mod.DoIt($mod.vP(1));',
  25232. '$mod.DoIt($mod.vP(2));',
  25233. '']));
  25234. end;
  25235. procedure TTestModule.TestProcType_ProcedureDelphi;
  25236. begin
  25237. StartProgram(false);
  25238. Add('{$mode Delphi}');
  25239. Add('type');
  25240. Add(' TProc = procedure;');
  25241. Add('procedure DoIt;');
  25242. Add('begin end;');
  25243. Add('var');
  25244. Add(' b: boolean;');
  25245. Add(' vP, vQ: tproc;');
  25246. Add('begin');
  25247. Add(' vp:=nil;');
  25248. Add(' vp:=vp;');
  25249. Add(' vp:=vq;');
  25250. Add(' vp:=@doit;'); // ok in fpc and delphi, Note that in Delphi type of @F is Pointer, while in FPC it is the proc type
  25251. Add(' vp:=doit;'); // illegal in fpc, ok in delphi
  25252. //Add(' vp:=@doit;'); // illegal in fpc, ok in delphi (because Delphi treats @F as Pointer), not supported by resolver
  25253. Add(' vp;'); // ok in fpc and delphi
  25254. Add(' vp();');
  25255. // equal
  25256. //Add(' b:=vp=nil;'); // ok in fpc, illegal in delphi
  25257. Add(' b:=@@vp=nil;'); // ok in fpc delphi mode, ok in delphi
  25258. //Add(' b:=nil=vp;'); // ok in fpc, illegal in delphi
  25259. Add(' b:=nil=@@vp;'); // ok in fpc delphi mode, ok in delphi
  25260. Add(' b:=@@vp=@@vq;'); // ok in fpc delphi mode, ok in Delphi
  25261. //Add(' b:=vp=vq;'); // in fpc compare proctypes, in delphi compare results
  25262. //Add(' b:=vp=@doit;'); // ok in fpc, illegal in delphi
  25263. Add(' b:=@@vp=@doit;'); // ok in fpc delphi mode, ok in delphi
  25264. //Add(' b:=@doit=vp;'); // ok in fpc, illegal in delphi
  25265. Add(' b:=@doit=@@vp;'); // ok in fpc delphi mode, ok in delphi
  25266. // unequal
  25267. //Add(' b:=vp<>nil;'); // ok in fpc, illegal in delphi
  25268. Add(' b:=@@vp<>nil;'); // ok in fpc mode delphi, ok in delphi
  25269. //Add(' b:=nil<>vp;'); // ok in fpc, illegal in delphi
  25270. Add(' b:=nil<>@@vp;'); // ok in fpc mode delphi, ok in delphi
  25271. //Add(' b:=vp<>vq;'); // in fpc compare proctypes, in delphi compare results
  25272. Add(' b:=@@vp<>@@vq;'); // ok in fpc mode delphi, ok in delphi
  25273. //Add(' b:=vp<>@doit;'); // ok in fpc, illegal in delphi
  25274. Add(' b:=@@vp<>@doit;'); // ok in fpc mode delphi, illegal in delphi
  25275. //Add(' b:=@doit<>vp;'); // ok in fpc, illegal in delphi
  25276. Add(' b:=@doit<>@@vp;'); // ok in fpc mode delphi, illegal in delphi
  25277. Add(' b:=Assigned(vp);');
  25278. ConvertProgram;
  25279. CheckSource('TestProcType_ProcedureDelphi',
  25280. LinesToStr([ // statements
  25281. 'this.DoIt = function() {',
  25282. '};',
  25283. 'this.b = false;',
  25284. 'this.vP = null;',
  25285. 'this.vQ = null;'
  25286. ]),
  25287. LinesToStr([ // $mod.$main
  25288. '$mod.vP = null;',
  25289. '$mod.vP = $mod.vP;',
  25290. '$mod.vP = $mod.vQ;',
  25291. '$mod.vP = $mod.DoIt;',
  25292. '$mod.vP = $mod.DoIt;',
  25293. '$mod.vP();',
  25294. '$mod.vP();',
  25295. '$mod.b = $mod.vP === null;',
  25296. '$mod.b = null === $mod.vP;',
  25297. '$mod.b = rtl.eqCallback($mod.vP, $mod.vQ);',
  25298. '$mod.b = rtl.eqCallback($mod.vP, $mod.DoIt);',
  25299. '$mod.b = rtl.eqCallback($mod.DoIt, $mod.vP);',
  25300. '$mod.b = $mod.vP !== null;',
  25301. '$mod.b = null !== $mod.vP;',
  25302. '$mod.b = !rtl.eqCallback($mod.vP, $mod.vQ);',
  25303. '$mod.b = !rtl.eqCallback($mod.vP, $mod.DoIt);',
  25304. '$mod.b = !rtl.eqCallback($mod.DoIt, $mod.vP);',
  25305. '$mod.b = $mod.vP != null;',
  25306. '']));
  25307. end;
  25308. procedure TTestModule.TestProcType_AsParam;
  25309. begin
  25310. StartProgram(false);
  25311. Add('type');
  25312. Add(' TFuncInt = function(vA: longint = 1): longint;');
  25313. Add('procedure DoIt(vG: tfuncint; const vH: tfuncint; var vI: tfuncint);');
  25314. Add('var vJ: tfuncint;');
  25315. Add('begin');
  25316. Add(' vg:=vg;');
  25317. Add(' vj:=vh;');
  25318. Add(' vi:=vi;');
  25319. Add(' doit(vg,vg,vg);');
  25320. Add(' doit(vh,vh,vj);');
  25321. Add(' doit(vi,vi,vi);');
  25322. Add(' doit(vj,vj,vj);');
  25323. Add('end;');
  25324. Add('var i: tfuncint;');
  25325. Add('begin');
  25326. Add(' doit(i,i,i);');
  25327. ConvertProgram;
  25328. CheckSource('TestProcType_AsParam',
  25329. LinesToStr([ // statements
  25330. 'this.DoIt = function (vG,vH,vI) {',
  25331. ' var vJ = null;',
  25332. ' vG = vG;',
  25333. ' vJ = vH;',
  25334. ' vI.set(vI.get());',
  25335. ' $mod.DoIt(vG, vG, {',
  25336. ' get: function () {',
  25337. ' return vG;',
  25338. ' },',
  25339. ' set: function (v) {',
  25340. ' vG = v;',
  25341. ' }',
  25342. ' });',
  25343. ' $mod.DoIt(vH, vH, {',
  25344. ' get: function () {',
  25345. ' return vJ;',
  25346. ' },',
  25347. ' set: function (v) {',
  25348. ' vJ = v;',
  25349. ' }',
  25350. ' });',
  25351. ' $mod.DoIt(vI.get(), vI.get(), vI);',
  25352. ' $mod.DoIt(vJ, vJ, {',
  25353. ' get: function () {',
  25354. ' return vJ;',
  25355. ' },',
  25356. ' set: function (v) {',
  25357. ' vJ = v;',
  25358. ' }',
  25359. ' });',
  25360. '};',
  25361. 'this.i = null;'
  25362. ]),
  25363. LinesToStr([
  25364. '$mod.DoIt($mod.i,$mod.i,{',
  25365. ' p: $mod,',
  25366. ' get: function () {',
  25367. ' return this.p.i;',
  25368. ' },',
  25369. ' set: function (v) {',
  25370. ' this.p.i = v;',
  25371. ' }',
  25372. '});'
  25373. ]));
  25374. end;
  25375. procedure TTestModule.TestProcType_MethodFPC;
  25376. begin
  25377. StartProgram(false);
  25378. Add('type');
  25379. Add(' TFuncInt = function(vA: longint = 1): longint of object;');
  25380. Add(' TObject = class');
  25381. Add(' function DoIt(vA: longint = 1): longint;');
  25382. Add(' end;');
  25383. Add('function TObject.DoIt(vA: longint = 1): longint;');
  25384. Add('begin');
  25385. Add('end;');
  25386. Add('var');
  25387. Add(' Obj: TObject;');
  25388. Add(' vP: tfuncint;');
  25389. Add(' b: boolean;');
  25390. Add('begin');
  25391. Add(' vp:[email protected];'); // ok in fpc and delphi
  25392. //Add(' vp:=obj.doit;'); // illegal in fpc, ok in delphi
  25393. Add(' vp;'); // ok in fpc and delphi
  25394. Add(' vp();');
  25395. Add(' vp(2);');
  25396. Add(' b:[email protected];'); // ok in fpc, illegal in delphi
  25397. Add(' b:[email protected]=vp;'); // ok in fpc, illegal in delphi
  25398. Add(' b:=vp<>@obj.doit;'); // ok in fpc, illegal in delphi
  25399. Add(' b:[email protected]<>vp;'); // ok in fpc, illegal in delphi
  25400. ConvertProgram;
  25401. CheckSource('TestProcType_MethodFPC',
  25402. LinesToStr([ // statements
  25403. 'rtl.createClass(this, "TObject", null, function () {',
  25404. ' this.$init = function () {',
  25405. ' };',
  25406. ' this.$final = function () {',
  25407. ' };',
  25408. ' this.DoIt = function (vA) {',
  25409. ' var Result = 0;',
  25410. ' return Result;',
  25411. ' };',
  25412. '});',
  25413. 'this.Obj = null;',
  25414. 'this.vP = null;',
  25415. 'this.b = false;'
  25416. ]),
  25417. LinesToStr([
  25418. '$mod.vP = rtl.createCallback($mod.Obj, "DoIt");',
  25419. '$mod.vP(1);',
  25420. '$mod.vP(1);',
  25421. '$mod.vP(2);',
  25422. '$mod.b = rtl.eqCallback($mod.vP, rtl.createCallback($mod.Obj, "DoIt"));',
  25423. '$mod.b = rtl.eqCallback(rtl.createCallback($mod.Obj, "DoIt"), $mod.vP);',
  25424. '$mod.b = !rtl.eqCallback($mod.vP, rtl.createCallback($mod.Obj, "DoIt"));',
  25425. '$mod.b = !rtl.eqCallback(rtl.createCallback($mod.Obj, "DoIt"), $mod.vP);',
  25426. '']));
  25427. end;
  25428. procedure TTestModule.TestProcType_MethodDelphi;
  25429. begin
  25430. StartProgram(false);
  25431. Add([
  25432. '{$mode delphi}',
  25433. 'type',
  25434. ' TFuncInt = function(vA: longint = 1): longint of object;',
  25435. ' TObject = class',
  25436. ' function DoIt(vA: longint = 1): longint;',
  25437. ' end;',
  25438. 'function TObject.DoIt(vA: longint = 1): longint;',
  25439. 'begin',
  25440. 'end;',
  25441. 'var',
  25442. ' Obj: TObject;',
  25443. ' vP: tfuncint;',
  25444. ' b: boolean;',
  25445. 'begin',
  25446. ' vp:[email protected];', // ok in fpc and delphi
  25447. ' vp:=obj.doit;', // illegal in fpc, ok in delphi
  25448. ' vp;', // ok in fpc and delphi
  25449. ' vp();',
  25450. ' vp(2);',
  25451. //' b:[email protected];', // ok in fpc, illegal in delphi
  25452. //' b:[email protected]=vp;', // ok in fpc, illegal in delphi
  25453. //' b:=vp<>@obj.doit;', // ok in fpc, illegal in delphi
  25454. //' b:[email protected]<>vp;'); // ok in fpc, illegal in delphi
  25455. '']);
  25456. ConvertProgram;
  25457. CheckSource('TestProcType_MethodDelphi',
  25458. LinesToStr([ // statements
  25459. 'rtl.createClass(this, "TObject", null, function () {',
  25460. ' this.$init = function () {',
  25461. ' };',
  25462. ' this.$final = function () {',
  25463. ' };',
  25464. ' this.DoIt = function (vA) {',
  25465. ' var Result = 0;',
  25466. ' return Result;',
  25467. ' };',
  25468. '});',
  25469. 'this.Obj = null;',
  25470. 'this.vP = null;',
  25471. 'this.b = false;'
  25472. ]),
  25473. LinesToStr([
  25474. '$mod.vP = rtl.createCallback($mod.Obj, "DoIt");',
  25475. '$mod.vP = rtl.createCallback($mod.Obj, "DoIt");',
  25476. '$mod.vP(1);',
  25477. '$mod.vP(1);',
  25478. '$mod.vP(2);',
  25479. '']));
  25480. end;
  25481. procedure TTestModule.TestProcType_PropertyFPC;
  25482. begin
  25483. StartProgram(false);
  25484. Add('type');
  25485. Add(' TFuncInt = function(vA: longint = 1): longint of object;');
  25486. Add(' TObject = class');
  25487. Add(' FOnFoo: TFuncInt;');
  25488. Add(' function DoIt(vA: longint = 1): longint;');
  25489. Add(' function GetFoo: TFuncInt;');
  25490. Add(' procedure SetFoo(const Value: TFuncInt);');
  25491. Add(' function GetEvents(Index: longint): TFuncInt;');
  25492. Add(' procedure SetEvents(Index: longint; const Value: TFuncInt);');
  25493. Add(' property OnFoo: TFuncInt read FOnFoo write FOnFoo;');
  25494. Add(' property OnBar: TFuncInt read GetFoo write SetFoo;');
  25495. Add(' property Events[Index: longint]: TFuncInt read GetEvents write SetEvents; default;');
  25496. Add(' end;');
  25497. Add('function tobject.doit(va: longint = 1): longint; begin end;');
  25498. Add('function tobject.getfoo: tfuncint; begin end;');
  25499. Add('procedure tobject.setfoo(const value: tfuncint); begin end;');
  25500. Add('function tobject.getevents(index: longint): tfuncint; begin end;');
  25501. Add('procedure tobject.setevents(index: longint; const value: tfuncint); begin end;');
  25502. Add('var');
  25503. Add(' Obj: TObject;');
  25504. Add(' vP: tfuncint;');
  25505. Add(' b: boolean;');
  25506. Add('begin');
  25507. Add(' obj.onfoo:=nil;');
  25508. Add(' obj.onbar:=nil;');
  25509. Add(' obj.events[1]:=nil;');
  25510. Add(' obj.onfoo:=obj.onfoo;');
  25511. Add(' obj.onbar:=obj.onbar;');
  25512. Add(' obj.events[2]:=obj.events[3];');
  25513. Add(' obj.onfoo:[email protected];');
  25514. Add(' obj.onbar:[email protected];');
  25515. Add(' obj.events[4]:[email protected];');
  25516. //Add(' obj.onfoo:=obj.doit;'); // delphi
  25517. //Add(' obj.onbar:=obj.doit;'); // delphi
  25518. //Add(' obj.events[4]:=obj.doit;'); // delphi
  25519. Add(' obj.onfoo;');
  25520. Add(' obj.onbar;');
  25521. //Add(' obj.events[5];'); ToDo in pasresolver
  25522. Add(' obj.onfoo();');
  25523. Add(' obj.onbar();');
  25524. Add(' obj.events[6]();');
  25525. Add(' b:=obj.onfoo=nil;');
  25526. Add(' b:=obj.onbar=nil;');
  25527. Add(' b:=obj.events[7]=nil;');
  25528. Add(' b:=obj.onfoo<>nil;');
  25529. Add(' b:=obj.onbar<>nil;');
  25530. Add(' b:=obj.events[8]<>nil;');
  25531. Add(' b:=obj.onfoo=vp;');
  25532. Add(' b:=obj.onbar=vp;');
  25533. Add(' b:=obj.events[9]=vp;');
  25534. Add(' b:=obj.onfoo=obj.onfoo;');
  25535. Add(' b:=obj.onbar=obj.onfoo;');
  25536. Add(' b:=obj.events[10]=obj.onfoo;');
  25537. Add(' b:=obj.onfoo<>obj.onfoo;');
  25538. Add(' b:=obj.onbar<>obj.onfoo;');
  25539. Add(' b:=obj.events[11]<>obj.onfoo;');
  25540. Add(' b:[email protected];');
  25541. Add(' b:[email protected];');
  25542. Add(' b:=obj.events[12][email protected];');
  25543. Add(' b:=obj.onfoo<>@obj.doit;');
  25544. Add(' b:=obj.onbar<>@obj.doit;');
  25545. Add(' b:=obj.events[12]<>@obj.doit;');
  25546. Add(' b:=Assigned(obj.onfoo);');
  25547. Add(' b:=Assigned(obj.onbar);');
  25548. Add(' b:=Assigned(obj.events[13]);');
  25549. ConvertProgram;
  25550. CheckSource('TestProcType_PropertyFPC',
  25551. LinesToStr([ // statements
  25552. 'rtl.createClass(this, "TObject", null, function () {',
  25553. ' this.$init = function () {',
  25554. ' this.FOnFoo = null;',
  25555. ' };',
  25556. ' this.$final = function () {',
  25557. ' this.FOnFoo = undefined;',
  25558. ' };',
  25559. ' this.DoIt = function (vA) {',
  25560. ' var Result = 0;',
  25561. ' return Result;',
  25562. ' };',
  25563. 'this.GetFoo = function () {',
  25564. ' var Result = null;',
  25565. ' return Result;',
  25566. '};',
  25567. 'this.SetFoo = function (Value) {',
  25568. '};',
  25569. 'this.GetEvents = function (Index) {',
  25570. ' var Result = null;',
  25571. ' return Result;',
  25572. '};',
  25573. 'this.SetEvents = function (Index, Value) {',
  25574. '};',
  25575. '});',
  25576. 'this.Obj = null;',
  25577. 'this.vP = null;',
  25578. 'this.b = false;'
  25579. ]),
  25580. LinesToStr([
  25581. '$mod.Obj.FOnFoo = null;',
  25582. '$mod.Obj.SetFoo(null);',
  25583. '$mod.Obj.SetEvents(1, null);',
  25584. '$mod.Obj.FOnFoo = $mod.Obj.FOnFoo;',
  25585. '$mod.Obj.SetFoo($mod.Obj.GetFoo());',
  25586. '$mod.Obj.SetEvents(2, $mod.Obj.GetEvents(3));',
  25587. '$mod.Obj.FOnFoo = rtl.createCallback($mod.Obj, "DoIt");',
  25588. '$mod.Obj.SetFoo(rtl.createCallback($mod.Obj, "DoIt"));',
  25589. '$mod.Obj.SetEvents(4, rtl.createCallback($mod.Obj, "DoIt"));',
  25590. '$mod.Obj.FOnFoo(1);',
  25591. '$mod.Obj.GetFoo();',
  25592. '$mod.Obj.FOnFoo(1);',
  25593. '$mod.Obj.GetFoo()(1);',
  25594. '$mod.Obj.GetEvents(6)(1);',
  25595. '$mod.b = $mod.Obj.FOnFoo === null;',
  25596. '$mod.b = $mod.Obj.GetFoo() === null;',
  25597. '$mod.b = $mod.Obj.GetEvents(7) === null;',
  25598. '$mod.b = $mod.Obj.FOnFoo !== null;',
  25599. '$mod.b = $mod.Obj.GetFoo() !== null;',
  25600. '$mod.b = $mod.Obj.GetEvents(8) !== null;',
  25601. '$mod.b = rtl.eqCallback($mod.Obj.FOnFoo, $mod.vP);',
  25602. '$mod.b = rtl.eqCallback($mod.Obj.GetFoo(), $mod.vP);',
  25603. '$mod.b = rtl.eqCallback($mod.Obj.GetEvents(9), $mod.vP);',
  25604. '$mod.b = rtl.eqCallback($mod.Obj.FOnFoo, $mod.Obj.FOnFoo);',
  25605. '$mod.b = rtl.eqCallback($mod.Obj.GetFoo(), $mod.Obj.FOnFoo);',
  25606. '$mod.b = rtl.eqCallback($mod.Obj.GetEvents(10), $mod.Obj.FOnFoo);',
  25607. '$mod.b = !rtl.eqCallback($mod.Obj.FOnFoo, $mod.Obj.FOnFoo);',
  25608. '$mod.b = !rtl.eqCallback($mod.Obj.GetFoo(), $mod.Obj.FOnFoo);',
  25609. '$mod.b = !rtl.eqCallback($mod.Obj.GetEvents(11), $mod.Obj.FOnFoo);',
  25610. '$mod.b = rtl.eqCallback($mod.Obj.FOnFoo, rtl.createCallback($mod.Obj, "DoIt"));',
  25611. '$mod.b = rtl.eqCallback($mod.Obj.GetFoo(), rtl.createCallback($mod.Obj, "DoIt"));',
  25612. '$mod.b = rtl.eqCallback($mod.Obj.GetEvents(12), rtl.createCallback($mod.Obj, "DoIt"));',
  25613. '$mod.b = !rtl.eqCallback($mod.Obj.FOnFoo, rtl.createCallback($mod.Obj, "DoIt"));',
  25614. '$mod.b = !rtl.eqCallback($mod.Obj.GetFoo(), rtl.createCallback($mod.Obj, "DoIt"));',
  25615. '$mod.b = !rtl.eqCallback($mod.Obj.GetEvents(12), rtl.createCallback($mod.Obj, "DoIt"));',
  25616. '$mod.b = $mod.Obj.FOnFoo != null;',
  25617. '$mod.b = $mod.Obj.GetFoo() != null;',
  25618. '$mod.b = $mod.Obj.GetEvents(13) != null;',
  25619. '']));
  25620. end;
  25621. procedure TTestModule.TestProcType_PropertyDelphi;
  25622. begin
  25623. StartProgram(false);
  25624. Add('{$mode delphi}');
  25625. Add('type');
  25626. Add(' TFuncInt = function(vA: longint = 1): longint of object;');
  25627. Add(' TObject = class');
  25628. Add(' FOnFoo: TFuncInt;');
  25629. Add(' function DoIt(vA: longint = 1): longint;');
  25630. Add(' function GetFoo: TFuncInt;');
  25631. Add(' procedure SetFoo(const Value: TFuncInt);');
  25632. Add(' function GetEvents(Index: longint): TFuncInt;');
  25633. Add(' procedure SetEvents(Index: longint; const Value: TFuncInt);');
  25634. Add(' property OnFoo: TFuncInt read FOnFoo write FOnFoo;');
  25635. Add(' property OnBar: TFuncInt read GetFoo write SetFoo;');
  25636. Add(' property Events[Index: longint]: TFuncInt read GetEvents write SetEvents; default;');
  25637. Add(' end;');
  25638. Add('function tobject.doit(va: longint = 1): longint; begin end;');
  25639. Add('function tobject.getfoo: tfuncint; begin end;');
  25640. Add('procedure tobject.setfoo(const value: tfuncint); begin end;');
  25641. Add('function tobject.getevents(index: longint): tfuncint; begin end;');
  25642. Add('procedure tobject.setevents(index: longint; const value: tfuncint); begin end;');
  25643. Add('var');
  25644. Add(' Obj: TObject;');
  25645. Add(' vP: tfuncint;');
  25646. Add(' b: boolean;');
  25647. Add('begin');
  25648. Add(' obj.onfoo:=nil;');
  25649. Add(' obj.onbar:=nil;');
  25650. Add(' obj.events[1]:=nil;');
  25651. Add(' obj.onfoo:=obj.onfoo;');
  25652. Add(' obj.onbar:=obj.onbar;');
  25653. Add(' obj.events[2]:=obj.events[3];');
  25654. Add(' obj.onfoo:[email protected];');
  25655. Add(' obj.onbar:[email protected];');
  25656. Add(' obj.events[4]:[email protected];');
  25657. Add(' obj.onfoo:=obj.doit;'); // delphi
  25658. Add(' obj.onbar:=obj.doit;'); // delphi
  25659. Add(' obj.events[4]:=obj.doit;'); // delphi
  25660. Add(' obj.onfoo;');
  25661. Add(' obj.onbar;');
  25662. //Add(' obj.events[5];'); ToDo in pasresolver
  25663. Add(' obj.onfoo();');
  25664. Add(' obj.onbar();');
  25665. Add(' obj.events[6]();');
  25666. //Add(' b:=obj.onfoo=nil;'); // fpc
  25667. //Add(' b:=obj.onbar=nil;'); // fpc
  25668. //Add(' b:=obj.events[7]=nil;'); // fpc
  25669. //Add(' b:=obj.onfoo<>nil;'); // fpc
  25670. //Add(' b:=obj.onbar<>nil;'); // fpc
  25671. //Add(' b:=obj.events[8]<>nil;'); // fpc
  25672. Add(' b:=obj.onfoo=vp;');
  25673. Add(' b:=obj.onbar=vp;');
  25674. //Add(' b:=obj.events[9]=vp;'); ToDo in pasresolver
  25675. Add(' b:=obj.onfoo=obj.onfoo;');
  25676. Add(' b:=obj.onbar=obj.onfoo;');
  25677. //Add(' b:=obj.events[10]=obj.onfoo;'); // ToDo in pasresolver
  25678. Add(' b:=obj.onfoo<>obj.onfoo;');
  25679. Add(' b:=obj.onbar<>obj.onfoo;');
  25680. //Add(' b:=obj.events[11]<>obj.onfoo;'); // ToDo in pasresolver
  25681. //Add(' b:[email protected];'); // fpc
  25682. //Add(' b:[email protected];'); // fpc
  25683. //Add(' b:=obj.events[12][email protected];'); // fpc
  25684. //Add(' b:=obj.onfoo<>@obj.doit;'); // fpc
  25685. //Add(' b:=obj.onbar<>@obj.doit;'); // fpc
  25686. //Add(' b:=obj.events[12]<>@obj.doit;'); // fpc
  25687. Add(' b:=Assigned(obj.onfoo);');
  25688. Add(' b:=Assigned(obj.onbar);');
  25689. Add(' b:=Assigned(obj.events[13]);');
  25690. ConvertProgram;
  25691. CheckSource('TestProcType_PropertyDelphi',
  25692. LinesToStr([ // statements
  25693. 'rtl.createClass(this, "TObject", null, function () {',
  25694. ' this.$init = function () {',
  25695. ' this.FOnFoo = null;',
  25696. ' };',
  25697. ' this.$final = function () {',
  25698. ' this.FOnFoo = undefined;',
  25699. ' };',
  25700. ' this.DoIt = function (vA) {',
  25701. ' var Result = 0;',
  25702. ' return Result;',
  25703. ' };',
  25704. 'this.GetFoo = function () {',
  25705. ' var Result = null;',
  25706. ' return Result;',
  25707. '};',
  25708. 'this.SetFoo = function (Value) {',
  25709. '};',
  25710. 'this.GetEvents = function (Index) {',
  25711. ' var Result = null;',
  25712. ' return Result;',
  25713. '};',
  25714. 'this.SetEvents = function (Index, Value) {',
  25715. '};',
  25716. '});',
  25717. 'this.Obj = null;',
  25718. 'this.vP = null;',
  25719. 'this.b = false;'
  25720. ]),
  25721. LinesToStr([
  25722. '$mod.Obj.FOnFoo = null;',
  25723. '$mod.Obj.SetFoo(null);',
  25724. '$mod.Obj.SetEvents(1, null);',
  25725. '$mod.Obj.FOnFoo = $mod.Obj.FOnFoo;',
  25726. '$mod.Obj.SetFoo($mod.Obj.GetFoo());',
  25727. '$mod.Obj.SetEvents(2, $mod.Obj.GetEvents(3));',
  25728. '$mod.Obj.FOnFoo = rtl.createCallback($mod.Obj, "DoIt");',
  25729. '$mod.Obj.SetFoo(rtl.createCallback($mod.Obj, "DoIt"));',
  25730. '$mod.Obj.SetEvents(4, rtl.createCallback($mod.Obj, "DoIt"));',
  25731. '$mod.Obj.FOnFoo = rtl.createCallback($mod.Obj, "DoIt");',
  25732. '$mod.Obj.SetFoo(rtl.createCallback($mod.Obj, "DoIt"));',
  25733. '$mod.Obj.SetEvents(4, rtl.createCallback($mod.Obj, "DoIt"));',
  25734. '$mod.Obj.FOnFoo(1);',
  25735. '$mod.Obj.GetFoo();',
  25736. '$mod.Obj.FOnFoo(1);',
  25737. '$mod.Obj.GetFoo()(1);',
  25738. '$mod.Obj.GetEvents(6)(1);',
  25739. '$mod.b = $mod.Obj.FOnFoo(1) === $mod.vP(1);',
  25740. '$mod.b = $mod.Obj.GetFoo() === $mod.vP(1);',
  25741. '$mod.b = $mod.Obj.FOnFoo(1) === $mod.Obj.FOnFoo(1);',
  25742. '$mod.b = $mod.Obj.GetFoo() === $mod.Obj.FOnFoo(1);',
  25743. '$mod.b = $mod.Obj.FOnFoo(1) !== $mod.Obj.FOnFoo(1);',
  25744. '$mod.b = $mod.Obj.GetFoo() !== $mod.Obj.FOnFoo(1);',
  25745. '$mod.b = $mod.Obj.FOnFoo != null;',
  25746. '$mod.b = $mod.Obj.GetFoo() != null;',
  25747. '$mod.b = $mod.Obj.GetEvents(13) != null;',
  25748. '']));
  25749. end;
  25750. procedure TTestModule.TestProcType_WithClassInstDoPropertyFPC;
  25751. begin
  25752. StartProgram(false);
  25753. Add('type');
  25754. Add(' TFuncInt = function(vA: longint = 1): longint of object;');
  25755. Add(' TObject = class');
  25756. Add(' FOnFoo: TFuncInt;');
  25757. Add(' function DoIt(vA: longint = 1): longint;');
  25758. Add(' function GetFoo: TFuncInt;');
  25759. Add(' procedure SetFoo(const Value: TFuncInt);');
  25760. Add(' property OnFoo: TFuncInt read FOnFoo write FOnFoo;');
  25761. Add(' property OnBar: TFuncInt read GetFoo write SetFoo;');
  25762. Add(' end;');
  25763. Add('function tobject.doit(va: longint = 1): longint; begin end;');
  25764. Add('function tobject.getfoo: tfuncint; begin end;');
  25765. Add('procedure tobject.setfoo(const value: tfuncint); begin end;');
  25766. Add('var');
  25767. Add(' Obj: TObject;');
  25768. Add(' vP: tfuncint;');
  25769. Add(' b: boolean;');
  25770. Add('begin');
  25771. Add('with obj do begin');
  25772. Add(' fonfoo:=nil;');
  25773. Add(' onfoo:=nil;');
  25774. Add(' onbar:=nil;');
  25775. Add(' fonfoo:=fonfoo;');
  25776. Add(' onfoo:=onfoo;');
  25777. Add(' onbar:=onbar;');
  25778. Add(' fonfoo:=@doit;');
  25779. Add(' onfoo:=@doit;');
  25780. Add(' onbar:=@doit;');
  25781. //Add(' fonfoo:=doit;'); // delphi
  25782. //Add(' onfoo:=doit;'); // delphi
  25783. //Add(' onbar:=doit;'); // delphi
  25784. Add(' fonfoo;');
  25785. Add(' onfoo;');
  25786. Add(' onbar;');
  25787. Add(' fonfoo();');
  25788. Add(' onfoo();');
  25789. Add(' onbar();');
  25790. Add(' b:=fonfoo=nil;');
  25791. Add(' b:=onfoo=nil;');
  25792. Add(' b:=onbar=nil;');
  25793. Add(' b:=fonfoo<>nil;');
  25794. Add(' b:=onfoo<>nil;');
  25795. Add(' b:=onbar<>nil;');
  25796. Add(' b:=fonfoo=vp;');
  25797. Add(' b:=onfoo=vp;');
  25798. Add(' b:=onbar=vp;');
  25799. Add(' b:=fonfoo=fonfoo;');
  25800. Add(' b:=onfoo=onfoo;');
  25801. Add(' b:=onbar=onfoo;');
  25802. Add(' b:=fonfoo<>fonfoo;');
  25803. Add(' b:=onfoo<>onfoo;');
  25804. Add(' b:=onbar<>onfoo;');
  25805. Add(' b:=fonfoo=@doit;');
  25806. Add(' b:=onfoo=@doit;');
  25807. Add(' b:=onbar=@doit;');
  25808. Add(' b:=fonfoo<>@doit;');
  25809. Add(' b:=onfoo<>@doit;');
  25810. Add(' b:=onbar<>@doit;');
  25811. Add(' b:=Assigned(fonfoo);');
  25812. Add(' b:=Assigned(onfoo);');
  25813. Add(' b:=Assigned(onbar);');
  25814. Add('end;');
  25815. ConvertProgram;
  25816. CheckSource('TestProcType_WithClassInstDoPropertyFPC',
  25817. LinesToStr([ // statements
  25818. 'rtl.createClass(this, "TObject", null, function () {',
  25819. ' this.$init = function () {',
  25820. ' this.FOnFoo = null;',
  25821. ' };',
  25822. ' this.$final = function () {',
  25823. ' this.FOnFoo = undefined;',
  25824. ' };',
  25825. ' this.DoIt = function (vA) {',
  25826. ' var Result = 0;',
  25827. ' return Result;',
  25828. ' };',
  25829. ' this.GetFoo = function () {',
  25830. ' var Result = null;',
  25831. ' return Result;',
  25832. ' };',
  25833. ' this.SetFoo = function (Value) {',
  25834. ' };',
  25835. '});',
  25836. 'this.Obj = null;',
  25837. 'this.vP = null;',
  25838. 'this.b = false;'
  25839. ]),
  25840. LinesToStr([
  25841. 'var $with = $mod.Obj;',
  25842. '$with.FOnFoo = null;',
  25843. '$with.FOnFoo = null;',
  25844. '$with.SetFoo(null);',
  25845. '$with.FOnFoo = $with.FOnFoo;',
  25846. '$with.FOnFoo = $with.FOnFoo;',
  25847. '$with.SetFoo($with.GetFoo());',
  25848. '$with.FOnFoo = rtl.createCallback($with, "DoIt");',
  25849. '$with.FOnFoo = rtl.createCallback($with, "DoIt");',
  25850. '$with.SetFoo(rtl.createCallback($with, "DoIt"));',
  25851. '$with.FOnFoo(1);',
  25852. '$with.FOnFoo(1);',
  25853. '$with.GetFoo();',
  25854. '$with.FOnFoo(1);',
  25855. '$with.FOnFoo(1);',
  25856. '$with.GetFoo()(1);',
  25857. '$mod.b = $with.FOnFoo === null;',
  25858. '$mod.b = $with.FOnFoo === null;',
  25859. '$mod.b = $with.GetFoo() === null;',
  25860. '$mod.b = $with.FOnFoo !== null;',
  25861. '$mod.b = $with.FOnFoo !== null;',
  25862. '$mod.b = $with.GetFoo() !== null;',
  25863. '$mod.b = rtl.eqCallback($with.FOnFoo, $mod.vP);',
  25864. '$mod.b = rtl.eqCallback($with.FOnFoo, $mod.vP);',
  25865. '$mod.b = rtl.eqCallback($with.GetFoo(), $mod.vP);',
  25866. '$mod.b = rtl.eqCallback($with.FOnFoo, $with.FOnFoo);',
  25867. '$mod.b = rtl.eqCallback($with.FOnFoo, $with.FOnFoo);',
  25868. '$mod.b = rtl.eqCallback($with.GetFoo(), $with.FOnFoo);',
  25869. '$mod.b = !rtl.eqCallback($with.FOnFoo, $with.FOnFoo);',
  25870. '$mod.b = !rtl.eqCallback($with.FOnFoo, $with.FOnFoo);',
  25871. '$mod.b = !rtl.eqCallback($with.GetFoo(), $with.FOnFoo);',
  25872. '$mod.b = rtl.eqCallback($with.FOnFoo, rtl.createCallback($with, "DoIt"));',
  25873. '$mod.b = rtl.eqCallback($with.FOnFoo, rtl.createCallback($with, "DoIt"));',
  25874. '$mod.b = rtl.eqCallback($with.GetFoo(), rtl.createCallback($with, "DoIt"));',
  25875. '$mod.b = !rtl.eqCallback($with.FOnFoo, rtl.createCallback($with, "DoIt"));',
  25876. '$mod.b = !rtl.eqCallback($with.FOnFoo, rtl.createCallback($with, "DoIt"));',
  25877. '$mod.b = !rtl.eqCallback($with.GetFoo(), rtl.createCallback($with, "DoIt"));',
  25878. '$mod.b = $with.FOnFoo != null;',
  25879. '$mod.b = $with.FOnFoo != null;',
  25880. '$mod.b = $with.GetFoo() != null;',
  25881. '']));
  25882. end;
  25883. procedure TTestModule.TestProcType_Nested;
  25884. begin
  25885. StartProgram(false);
  25886. Add([
  25887. 'type',
  25888. ' TProcInt = procedure(vI: longint = 1);',
  25889. 'procedure DoIt(vJ: longint);',
  25890. 'var aProc: TProcInt;',
  25891. ' b: boolean;',
  25892. ' procedure Sub(vK: longint);',
  25893. ' var aSub: TProcInt;',
  25894. ' procedure SubSub(vK: longint);',
  25895. ' var aSubSub: TProcInt;',
  25896. ' begin;',
  25897. ' aProc:=@DoIt;',
  25898. ' aSub:=@DoIt;',
  25899. ' aSubSub:=@DoIt;',
  25900. ' aProc:=@Sub;',
  25901. ' aSub:=@Sub;',
  25902. ' aSubSub:=@Sub;',
  25903. ' aProc:=@SubSub;',
  25904. ' aSub:=@SubSub;',
  25905. ' aSubSub:=@SubSub;',
  25906. ' end;',
  25907. ' begin;',
  25908. ' end;',
  25909. 'begin;',
  25910. ' aProc:=@Sub;',
  25911. ' b:=aProc=@Sub;',
  25912. ' b:=@Sub=aProc;',
  25913. 'end;',
  25914. 'begin',
  25915. '']);
  25916. ConvertProgram;
  25917. CheckSource('TestProcType_Nested',
  25918. LinesToStr([ // statements
  25919. 'this.DoIt = function (vJ) {',
  25920. ' var aProc = null;',
  25921. ' var b = false;',
  25922. ' function Sub(vK) {',
  25923. ' var aSub = null;',
  25924. ' function SubSub(vK) {',
  25925. ' var aSubSub = null;',
  25926. ' aProc = $mod.DoIt;',
  25927. ' aSub = $mod.DoIt;',
  25928. ' aSubSub = $mod.DoIt;',
  25929. ' aProc = Sub;',
  25930. ' aSub = Sub;',
  25931. ' aSubSub = Sub;',
  25932. ' aProc = SubSub;',
  25933. ' aSub = SubSub;',
  25934. ' aSubSub = SubSub;',
  25935. ' };',
  25936. ' };',
  25937. ' aProc = Sub;',
  25938. ' b = rtl.eqCallback(aProc, Sub);',
  25939. ' b = rtl.eqCallback(Sub, aProc);',
  25940. '};',
  25941. '']),
  25942. LinesToStr([ // $mod.$main
  25943. '']));
  25944. end;
  25945. procedure TTestModule.TestProcType_NestedOfObject;
  25946. begin
  25947. StartProgram(false);
  25948. Add([
  25949. 'type',
  25950. ' TProcInt = procedure(vI: longint = 1) of object;',
  25951. ' TObject = class',
  25952. ' procedure DoIt(vJ: longint);',
  25953. ' end;',
  25954. 'procedure TObject.DoIt(vJ: longint);',
  25955. 'var aProc: TProcInt;',
  25956. ' b: boolean;',
  25957. ' procedure Sub(vK: longint);',
  25958. ' var aSub: TProcInt;',
  25959. ' procedure SubSub(vK: longint);',
  25960. ' var aSubSub: TProcInt;',
  25961. ' begin;',
  25962. ' aProc:=@DoIt;',
  25963. ' aSub:=@DoIt;',
  25964. ' aSubSub:=@DoIt;',
  25965. ' aProc:=@Sub;',
  25966. ' aSub:=@Sub;',
  25967. ' aSubSub:=@Sub;',
  25968. ' aProc:=@SubSub;',
  25969. ' aSub:=@SubSub;',
  25970. ' aSubSub:=@SubSub;',
  25971. ' end;',
  25972. ' begin;',
  25973. ' end;',
  25974. 'begin;',
  25975. ' aProc:=@Sub;',
  25976. ' b:=aProc=@Sub;',
  25977. ' b:=@Sub=aProc;',
  25978. 'end;',
  25979. 'begin',
  25980. '']);
  25981. ConvertProgram;
  25982. CheckSource('TestProcType_Nested',
  25983. LinesToStr([ // statements
  25984. 'rtl.createClass(this, "TObject", null, function () {',
  25985. ' this.$init = function () {',
  25986. ' };',
  25987. ' this.$final = function () {',
  25988. ' };',
  25989. ' this.DoIt = function (vJ) {',
  25990. ' var $Self = this;',
  25991. ' var aProc = null;',
  25992. ' var b = false;',
  25993. ' function Sub(vK) {',
  25994. ' var aSub = null;',
  25995. ' function SubSub(vK) {',
  25996. ' var aSubSub = null;',
  25997. ' aProc = rtl.createCallback($Self, "DoIt");',
  25998. ' aSub = rtl.createCallback($Self, "DoIt");',
  25999. ' aSubSub = rtl.createCallback($Self, "DoIt");',
  26000. ' aProc = Sub;',
  26001. ' aSub = Sub;',
  26002. ' aSubSub = Sub;',
  26003. ' aProc = SubSub;',
  26004. ' aSub = SubSub;',
  26005. ' aSubSub = SubSub;',
  26006. ' };',
  26007. ' };',
  26008. ' aProc = Sub;',
  26009. ' b = rtl.eqCallback(aProc, Sub);',
  26010. ' b = rtl.eqCallback(Sub, aProc);',
  26011. ' };',
  26012. '});',
  26013. '']),
  26014. LinesToStr([ // $mod.$main
  26015. '']));
  26016. end;
  26017. procedure TTestModule.TestProcType_ReferenceToProc;
  26018. begin
  26019. StartProgram(false);
  26020. Add([
  26021. 'type',
  26022. ' TProcRef = reference to procedure(i: longint = 0);',
  26023. ' TFuncRef = reference to function(i: longint = 0): longint;',
  26024. 'var',
  26025. ' p: TProcRef;',
  26026. ' f: TFuncRef;',
  26027. 'procedure DoIt(i: longint);',
  26028. 'begin',
  26029. 'end;',
  26030. 'function GetIt(i: longint): longint;',
  26031. 'begin',
  26032. ' p:=@DoIt;',
  26033. ' f:=@GetIt;',
  26034. ' f;',
  26035. ' f();',
  26036. ' f(1);',
  26037. 'end;',
  26038. 'begin',
  26039. ' p:=@DoIt;',
  26040. ' f:=@GetIt;',
  26041. ' f;',
  26042. ' f();',
  26043. ' f(1);',
  26044. ' p:=TProcRef(f);',
  26045. '']);
  26046. ConvertProgram;
  26047. CheckSource('TestProcType_ReferenceToProc',
  26048. LinesToStr([ // statements
  26049. 'this.p = null;',
  26050. 'this.f = null;',
  26051. 'this.DoIt = function (i) {',
  26052. '};',
  26053. 'this.GetIt = function (i) {',
  26054. ' var Result = 0;',
  26055. ' $mod.p = $mod.DoIt;',
  26056. ' $mod.f = $mod.GetIt;',
  26057. ' $mod.f(0);',
  26058. ' $mod.f(0);',
  26059. ' $mod.f(1);',
  26060. ' return Result;',
  26061. '};',
  26062. '']),
  26063. LinesToStr([ // $mod.$main
  26064. '$mod.p = $mod.DoIt;',
  26065. '$mod.f = $mod.GetIt;',
  26066. '$mod.f(0);',
  26067. '$mod.f(0);',
  26068. '$mod.f(1);',
  26069. '$mod.p = $mod.f;',
  26070. '']));
  26071. end;
  26072. procedure TTestModule.TestProcType_ReferenceToMethod;
  26073. begin
  26074. StartProgram(false);
  26075. Add([
  26076. 'type',
  26077. ' TFuncRef = reference to function(i: longint = 5): longint;',
  26078. ' TObject = class',
  26079. ' function Grow(s: longint): longint;',
  26080. ' end;',
  26081. 'var',
  26082. ' f: tfuncref;',
  26083. 'function tobject.grow(s: longint): longint;',
  26084. ' function GrowSub(i: longint): longint;',
  26085. ' begin',
  26086. ' f:=@grow;',
  26087. ' f:=@growsub;',
  26088. ' end;',
  26089. 'begin',
  26090. ' f:=@grow;',
  26091. ' f:=@growsub;',
  26092. 'end;',
  26093. 'begin',
  26094. '']);
  26095. ConvertProgram;
  26096. CheckSource('TestProcType_ReferenceToMethod',
  26097. LinesToStr([ // statements
  26098. 'rtl.createClass(this, "TObject", null, function () {',
  26099. ' this.$init = function () {',
  26100. ' };',
  26101. ' this.$final = function () {',
  26102. ' };',
  26103. ' this.Grow = function (s) {',
  26104. ' var $Self = this;',
  26105. ' var Result = 0;',
  26106. ' function GrowSub(i) {',
  26107. ' var Result = 0;',
  26108. ' $mod.f = rtl.createCallback($Self, "Grow");',
  26109. ' $mod.f = GrowSub;',
  26110. ' return Result;',
  26111. ' };',
  26112. ' $mod.f = rtl.createCallback($Self, "Grow");',
  26113. ' $mod.f = GrowSub;',
  26114. ' return Result;',
  26115. ' };',
  26116. '});',
  26117. 'this.f = null;',
  26118. '']),
  26119. LinesToStr([ // $mod.$main
  26120. '']));
  26121. end;
  26122. procedure TTestModule.TestProcType_Typecast;
  26123. begin
  26124. StartProgram(false);
  26125. Add([
  26126. 'type',
  26127. ' TNotifyEvent = procedure(Sender: Pointer) of object;',
  26128. ' TEvent = procedure of object;',
  26129. ' TGetter = function:longint of object;',
  26130. ' TProcA = procedure(i: longint);',
  26131. ' TFuncB = function(i, j: longint): longint;',
  26132. 'procedure DoIt(); varargs; begin end;',
  26133. 'var',
  26134. ' Notify: tnotifyevent;',
  26135. ' Event: tevent;',
  26136. ' Getter: tgetter;',
  26137. ' ProcA: tproca;',
  26138. ' FuncB: tfuncb;',
  26139. ' p: pointer;',
  26140. 'begin',
  26141. ' notify:=tnotifyevent(event);',
  26142. ' event:=tevent(event);',
  26143. ' event:=tevent(notify);',
  26144. ' event:=tevent(getter);',
  26145. ' event:=tevent(proca);',
  26146. ' proca:=tproca(funcb);',
  26147. ' funcb:=tfuncb(funcb);',
  26148. ' funcb:=tfuncb(proca);',
  26149. ' funcb:=tfuncb(getter);',
  26150. ' proca:=tproca(p);',
  26151. ' funcb:=tfuncb(p);',
  26152. ' getter:=tgetter(p);',
  26153. ' p:=pointer(notify);',
  26154. ' p:=notify;',
  26155. ' p:=pointer(proca);',
  26156. ' p:=proca;',
  26157. ' p:=pointer(funcb);',
  26158. ' p:=funcb;',
  26159. ' doit(Pointer(notify),pointer(event),pointer(proca));',
  26160. '']);
  26161. ConvertProgram;
  26162. CheckSource('TestProcType_Typecast',
  26163. LinesToStr([ // statements
  26164. 'this.DoIt = function () {',
  26165. '};',
  26166. 'this.Notify = null;',
  26167. 'this.Event = null;',
  26168. 'this.Getter = null;',
  26169. 'this.ProcA = null;',
  26170. 'this.FuncB = null;',
  26171. 'this.p = null;',
  26172. '']),
  26173. LinesToStr([ // $mod.$main
  26174. '$mod.Notify = $mod.Event;',
  26175. '$mod.Event = $mod.Event;',
  26176. '$mod.Event = $mod.Notify;',
  26177. '$mod.Event = $mod.Getter;',
  26178. '$mod.Event = $mod.ProcA;',
  26179. '$mod.ProcA = $mod.FuncB;',
  26180. '$mod.FuncB = $mod.FuncB;',
  26181. '$mod.FuncB = $mod.ProcA;',
  26182. '$mod.FuncB = $mod.Getter;',
  26183. '$mod.ProcA = $mod.p;',
  26184. '$mod.FuncB = $mod.p;',
  26185. '$mod.Getter = $mod.p;',
  26186. '$mod.p = $mod.Notify;',
  26187. '$mod.p = $mod.Notify;',
  26188. '$mod.p = $mod.ProcA;',
  26189. '$mod.p = $mod.ProcA;',
  26190. '$mod.p = $mod.FuncB;',
  26191. '$mod.p = $mod.FuncB;',
  26192. '$mod.DoIt($mod.Notify, $mod.Event, $mod.ProcA);',
  26193. '']));
  26194. end;
  26195. procedure TTestModule.TestProcType_PassProcToUntyped;
  26196. begin
  26197. StartProgram(false);
  26198. Add([
  26199. 'type',
  26200. ' TEvent = procedure of object;',
  26201. ' TFunc = function: longint;',
  26202. 'procedure DoIt(); varargs; begin end;',
  26203. 'procedure DoSome(const a; var b; p: pointer); begin end;',
  26204. 'var',
  26205. ' Event: tevent;',
  26206. ' Func: TFunc;',
  26207. 'begin',
  26208. ' doit(event,func);',
  26209. ' dosome(event,event,event);',
  26210. ' dosome(func,func,func);',
  26211. '']);
  26212. ConvertProgram;
  26213. CheckSource('TestProcType_PassProcToUntyped',
  26214. LinesToStr([ // statements
  26215. 'this.DoIt = function () {',
  26216. '};',
  26217. 'this.DoSome = function (a, b, p) {',
  26218. '};',
  26219. 'this.Event = null;',
  26220. 'this.Func = null;',
  26221. '']),
  26222. LinesToStr([ // $mod.$main
  26223. '$mod.DoIt($mod.Event, $mod.Func);',
  26224. '$mod.DoSome($mod.Event, {',
  26225. ' p: $mod,',
  26226. ' get: function () {',
  26227. ' return this.p.Event;',
  26228. ' },',
  26229. ' set: function (v) {',
  26230. ' this.p.Event = v;',
  26231. ' }',
  26232. '}, $mod.Event);',
  26233. '$mod.DoSome($mod.Func, {',
  26234. ' p: $mod,',
  26235. ' get: function () {',
  26236. ' return this.p.Func;',
  26237. ' },',
  26238. ' set: function (v) {',
  26239. ' this.p.Func = v;',
  26240. ' }',
  26241. '}, $mod.Func);',
  26242. '']));
  26243. end;
  26244. procedure TTestModule.TestProcType_PassProcToArray;
  26245. begin
  26246. StartProgram(false);
  26247. Add([
  26248. 'type',
  26249. ' TFunc = function: longint;',
  26250. ' TArrFunc = array of TFunc;',
  26251. 'procedure DoIt(Arr: TArrFunc); begin end;',
  26252. 'function GetIt: longint; begin end;',
  26253. 'var',
  26254. ' Func: tfunc;',
  26255. 'begin',
  26256. ' doit([]);',
  26257. ' doit([@GetIt]);',
  26258. ' doit([Func]);',
  26259. '']);
  26260. ConvertProgram;
  26261. CheckSource('TestProcType_PassProcToArray',
  26262. LinesToStr([ // statements
  26263. 'this.DoIt = function (Arr) {',
  26264. '};',
  26265. 'this.GetIt = function () {',
  26266. ' var Result = 0;',
  26267. ' return Result;',
  26268. '};',
  26269. 'this.Func = null;',
  26270. '']),
  26271. LinesToStr([ // $mod.$main
  26272. '$mod.DoIt([]);',
  26273. '$mod.DoIt([$mod.GetIt]);',
  26274. '$mod.DoIt([$mod.Func]);',
  26275. '']));
  26276. end;
  26277. procedure TTestModule.TestProcType_SafeCallObjFPC;
  26278. begin
  26279. StartProgram(false);
  26280. Add([
  26281. '{$modeswitch externalclass}',
  26282. 'type',
  26283. ' TProc = reference to procedure(i: longint); safecall;',
  26284. ' TEvent = procedure(i: longint) of object; safecall;',
  26285. ' TExtA = class external name ''ExtObj''',
  26286. ' procedure DoIt(Id: longint = 1); external name ''$Execute'';',
  26287. ' procedure DoSome(Id: longint = 1);',
  26288. ' procedure SetOnClick(const e: TEvent);',
  26289. ' property OnClick: TEvent write SetOnClick;',
  26290. ' class procedure Fly(Id: longint = 1); static;',
  26291. ' procedure SetOnShow(const p: TProc);',
  26292. ' property OnShow: TProc write SetOnShow;',
  26293. ' end;',
  26294. 'procedure Run(i: longint = 1);',
  26295. 'begin',
  26296. 'end;',
  26297. 'var',
  26298. ' Obj: texta;',
  26299. ' e: TEvent;',
  26300. ' p: TProc;',
  26301. 'begin',
  26302. ' e:=e;',
  26303. ' e:[email protected];',
  26304. ' e:[email protected];',
  26305. ' e:=TEvent(@obj.dosome);', // no safecall
  26306. ' obj.OnClick:[email protected];',
  26307. ' obj.OnClick:[email protected];',
  26308. ' obj.setonclick(@obj.doit);',
  26309. ' obj.setonclick(@obj.dosome);',
  26310. ' p:=@Run;',
  26311. ' p:[email protected];',
  26312. ' obj.OnShow:=@Run;',
  26313. ' obj.OnShow:[email protected];',
  26314. ' obj.setOnShow(@Run);',
  26315. ' obj.setOnShow(@TExtA.Fly);',
  26316. ' with obj do begin',
  26317. ' e:=@doit;',
  26318. ' e:=@dosome;',
  26319. ' OnClick:=@doit;',
  26320. ' OnClick:=@dosome;',
  26321. ' setonclick(@doit);',
  26322. ' setonclick(@dosome);',
  26323. ' OnShow:=@Run;',
  26324. ' setOnShow(@Run);',
  26325. ' end;']);
  26326. ConvertProgram;
  26327. CheckSource('TestProcType_SafeCallObjFPC',
  26328. LinesToStr([ // statements
  26329. 'this.Run = function (i) {',
  26330. '};',
  26331. 'this.Obj = null;',
  26332. 'this.e = null;',
  26333. 'this.p = null;',
  26334. '']),
  26335. LinesToStr([ // $mod.$main
  26336. '$mod.e = $mod.e;',
  26337. '$mod.e = rtl.createSafeCallback($mod.Obj, "$Execute");',
  26338. '$mod.e = rtl.createSafeCallback($mod.Obj, "DoSome");',
  26339. '$mod.e = rtl.createCallback($mod.Obj, "DoSome");',
  26340. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "$Execute"));',
  26341. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "DoSome"));',
  26342. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "$Execute"));',
  26343. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "DoSome"));',
  26344. '$mod.p = rtl.createSafeCallback($mod, "Run");',
  26345. '$mod.p = rtl.createSafeCallback(ExtObj, "Fly");',
  26346. '$mod.Obj.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26347. '$mod.Obj.SetOnShow(rtl.createSafeCallback(ExtObj, "Fly"));',
  26348. '$mod.Obj.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26349. '$mod.Obj.SetOnShow(rtl.createSafeCallback(ExtObj, "Fly"));',
  26350. 'var $with = $mod.Obj;',
  26351. '$mod.e = rtl.createSafeCallback($with, "$Execute");',
  26352. '$mod.e = rtl.createSafeCallback($with, "DoSome");',
  26353. '$with.SetOnClick(rtl.createSafeCallback($with, "$Execute"));',
  26354. '$with.SetOnClick(rtl.createSafeCallback($with, "DoSome"));',
  26355. '$with.SetOnClick(rtl.createSafeCallback($with, "$Execute"));',
  26356. '$with.SetOnClick(rtl.createSafeCallback($with, "DoSome"));',
  26357. '$with.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26358. '$with.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26359. '']));
  26360. end;
  26361. procedure TTestModule.TestProcType_SafeCallDelphi;
  26362. begin
  26363. StartProgram(false);
  26364. Add([
  26365. '{$mode delphi}',
  26366. '{$modeswitch externalclass}',
  26367. 'type',
  26368. ' TProc = reference to procedure(i: longint); safecall;',
  26369. ' TEvent = procedure(i: longint) of object; safecall;',
  26370. ' TExtA = class external name ''ExtObj''',
  26371. ' procedure DoIt(Id: longint = 1); external name ''$Execute'';',
  26372. ' procedure DoSome(Id: longint = 1);',
  26373. ' procedure SetOnClick(const e: TEvent);',
  26374. ' property OnClick: TEvent write SetOnClick;',
  26375. ' class procedure Fly(Id: longint = 1); static;',
  26376. ' procedure SetOnShow(const p: TProc);',
  26377. ' property OnShow: TProc write SetOnShow;',
  26378. ' end;',
  26379. 'procedure Run(i: longint = 1);',
  26380. 'begin',
  26381. 'end;',
  26382. 'var',
  26383. ' Obj: texta;',
  26384. ' e: TEvent;',
  26385. ' p: TProc;',
  26386. 'begin',
  26387. ' e:=e;',
  26388. ' e:=obj.doit;',
  26389. ' e:=obj.dosome;',
  26390. ' e:=TEvent(@obj.dosome);', // no safecall
  26391. ' obj.OnClick:=obj.doit;',
  26392. ' obj.OnClick:=obj.dosome;',
  26393. ' obj.setonclick(obj.doit);',
  26394. ' obj.setonclick(obj.dosome);',
  26395. ' p:=Run;',
  26396. ' p:=TExtA.Fly;',
  26397. ' obj.OnShow:=Run;',
  26398. ' obj.OnShow:=TExtA.Fly;',
  26399. ' obj.setOnShow(Run);',
  26400. ' obj.setOnShow(TExtA.Fly);',
  26401. ' with obj do begin',
  26402. ' e:=doit;',
  26403. ' e:=dosome;',
  26404. ' OnClick:=doit;',
  26405. ' OnClick:=dosome;',
  26406. ' setonclick(doit);',
  26407. ' setonclick(dosome);',
  26408. ' OnShow:=@Run;',
  26409. ' setOnShow(@Run);',
  26410. ' end;']);
  26411. ConvertProgram;
  26412. CheckSource('TestProcType_SafeCallDelphi',
  26413. LinesToStr([ // statements
  26414. 'this.Run = function (i) {',
  26415. '};',
  26416. 'this.Obj = null;',
  26417. 'this.e = null;',
  26418. 'this.p = null;',
  26419. '']),
  26420. LinesToStr([ // $mod.$main
  26421. '$mod.e = $mod.e;',
  26422. '$mod.e = rtl.createSafeCallback($mod.Obj, "$Execute");',
  26423. '$mod.e = rtl.createSafeCallback($mod.Obj, "DoSome");',
  26424. '$mod.e = rtl.createCallback($mod.Obj, "DoSome");',
  26425. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "$Execute"));',
  26426. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "DoSome"));',
  26427. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "$Execute"));',
  26428. '$mod.Obj.SetOnClick(rtl.createSafeCallback($mod.Obj, "DoSome"));',
  26429. '$mod.p = rtl.createSafeCallback($mod, "Run");',
  26430. '$mod.p = rtl.createSafeCallback(ExtObj, "Fly");',
  26431. '$mod.Obj.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26432. '$mod.Obj.SetOnShow(rtl.createSafeCallback(ExtObj, "Fly"));',
  26433. '$mod.Obj.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26434. '$mod.Obj.SetOnShow(rtl.createSafeCallback(ExtObj, "Fly"));',
  26435. 'var $with = $mod.Obj;',
  26436. '$mod.e = rtl.createSafeCallback($with, "$Execute");',
  26437. '$mod.e = rtl.createSafeCallback($with, "DoSome");',
  26438. '$with.SetOnClick(rtl.createSafeCallback($with, "$Execute"));',
  26439. '$with.SetOnClick(rtl.createSafeCallback($with, "DoSome"));',
  26440. '$with.SetOnClick(rtl.createSafeCallback($with, "$Execute"));',
  26441. '$with.SetOnClick(rtl.createSafeCallback($with, "DoSome"));',
  26442. '$with.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26443. '$with.SetOnShow(rtl.createSafeCallback($mod, "Run"));',
  26444. '']));
  26445. end;
  26446. procedure TTestModule.TestPointer;
  26447. begin
  26448. StartProgram(false);
  26449. Add(['type',
  26450. ' TObject = class end;',
  26451. ' TClass = class of TObject;',
  26452. ' TArrInt = array of longint;',
  26453. 'const',
  26454. ' n = nil;',
  26455. 'var',
  26456. ' v: jsvalue;',
  26457. ' Obj: tobject;',
  26458. ' C: tclass;',
  26459. ' a: tarrint;',
  26460. ' p: Pointer = nil;',
  26461. ' s: string;',
  26462. 'begin',
  26463. ' p:=p;',
  26464. ' p:=nil;',
  26465. ' if p=nil then;',
  26466. ' if nil=p then;',
  26467. ' if Assigned(p) then;',
  26468. ' p:=Pointer(v);',
  26469. ' p:=obj;',
  26470. ' p:=c;',
  26471. ' p:=a;',
  26472. ' p:=tobject;',
  26473. ' obj:=TObject(p);',
  26474. ' c:=TClass(p);',
  26475. ' a:=TArrInt(p);',
  26476. ' p:=n;',
  26477. ' p:=Pointer(a);',
  26478. ' p:=pointer(s);',
  26479. ' s:=string(p);',
  26480. '']);
  26481. ConvertProgram;
  26482. CheckSource('TestPointer',
  26483. LinesToStr([ // statements
  26484. 'rtl.createClass(this, "TObject", null, function () {',
  26485. ' this.$init = function () {',
  26486. ' };',
  26487. ' this.$final = function () {',
  26488. ' };',
  26489. '});',
  26490. 'this.n = null;',
  26491. 'this.v = undefined;',
  26492. 'this.Obj = null;',
  26493. 'this.C = null;',
  26494. 'this.a = [];',
  26495. 'this.p = null;',
  26496. 'this.s = "";',
  26497. '']),
  26498. LinesToStr([ // $mod.$main
  26499. '$mod.p = $mod.p;',
  26500. '$mod.p = null;',
  26501. 'if ($mod.p === null) ;',
  26502. 'if (null === $mod.p) ;',
  26503. 'if ($mod.p != null) ;',
  26504. '$mod.p = $mod.v;',
  26505. '$mod.p = $mod.Obj;',
  26506. '$mod.p = $mod.C;',
  26507. '$mod.p = $mod.a;',
  26508. '$mod.p = $mod.TObject;',
  26509. '$mod.Obj = $mod.p;',
  26510. '$mod.C = $mod.p;',
  26511. '$mod.a = $mod.p;',
  26512. '$mod.p = null;',
  26513. '$mod.p = $mod.a;',
  26514. '$mod.p = $mod.s;',
  26515. '$mod.s = $mod.p;',
  26516. '']));
  26517. end;
  26518. procedure TTestModule.TestPointer_Proc;
  26519. begin
  26520. StartProgram(false);
  26521. Add('type');
  26522. Add(' TObject = class');
  26523. Add(' procedure DoIt; virtual; abstract;');
  26524. Add(' end;');
  26525. Add('procedure DoSome; begin end;');
  26526. Add('var');
  26527. Add(' o: TObject;');
  26528. Add(' p: Pointer;');
  26529. Add('begin');
  26530. Add(' p:=@DoSome;');
  26531. Add(' p:[email protected];');
  26532. ConvertProgram;
  26533. CheckSource('TestPointer_Proc',
  26534. LinesToStr([ // statements
  26535. 'rtl.createClass(this, "TObject", null, function () {',
  26536. ' this.$init = function () {',
  26537. ' };',
  26538. ' this.$final = function () {',
  26539. ' };',
  26540. '});',
  26541. 'this.DoSome = function () {',
  26542. '};',
  26543. 'this.o = null;',
  26544. 'this.p = null;',
  26545. '']),
  26546. LinesToStr([ // $mod.$main
  26547. '$mod.p = $mod.DoSome;',
  26548. '$mod.p = rtl.createCallback($mod.o, "DoIt");',
  26549. '']));
  26550. end;
  26551. procedure TTestModule.TestPointer_AssignRecordFail;
  26552. begin
  26553. StartProgram(false);
  26554. Add('type');
  26555. Add(' TRec = record end;');
  26556. Add('var');
  26557. Add(' p: Pointer;');
  26558. Add(' r: TRec;');
  26559. Add('begin');
  26560. Add(' p:=r;');
  26561. SetExpectedPasResolverError('Incompatible types: got "TRec" expected "Pointer"',
  26562. nIncompatibleTypesGotExpected);
  26563. ConvertProgram;
  26564. end;
  26565. procedure TTestModule.TestPointer_AssignStaticArrayFail;
  26566. begin
  26567. StartProgram(false);
  26568. Add('type');
  26569. Add(' TArr = array[boolean] of longint;');
  26570. Add('var');
  26571. Add(' p: Pointer;');
  26572. Add(' a: TArr;');
  26573. Add('begin');
  26574. Add(' p:=a;');
  26575. SetExpectedPasResolverError('Incompatible types: got "TArr" expected "Pointer"',
  26576. nIncompatibleTypesGotExpected);
  26577. ConvertProgram;
  26578. end;
  26579. procedure TTestModule.TestPointer_TypeCastJSValueToPointer;
  26580. begin
  26581. StartProgram(false);
  26582. Add([
  26583. 'procedure DoIt(args: array of jsvalue); begin end;',
  26584. 'procedure DoAll; varargs; begin end;',
  26585. 'var',
  26586. ' v: jsvalue;',
  26587. 'begin',
  26588. ' DoIt([pointer(v)]);',
  26589. ' DoAll(pointer(v));',
  26590. '']);
  26591. ConvertProgram;
  26592. CheckSource('TestPointer_TypeCastJSValueToPointer',
  26593. LinesToStr([ // statements
  26594. 'this.DoIt = function (args) {',
  26595. '};',
  26596. 'this.DoAll = function () {',
  26597. '};',
  26598. 'this.v = undefined;',
  26599. '']),
  26600. LinesToStr([ // $mod.$main
  26601. '$mod.DoIt([$mod.v]);',
  26602. '$mod.DoAll($mod.v);',
  26603. '']));
  26604. end;
  26605. procedure TTestModule.TestPointer_NonRecordFail;
  26606. begin
  26607. StartProgram(false);
  26608. Add([
  26609. 'type',
  26610. ' p = ^longint;',
  26611. 'begin',
  26612. '']);
  26613. SetExpectedPasResolverError('Not supported: pointer of Longint',nNotSupportedX);
  26614. ConvertProgram;
  26615. end;
  26616. procedure TTestModule.TestPointer_AnonymousArgTypeFail;
  26617. begin
  26618. StartProgram(false);
  26619. Add([
  26620. 'procedure DoIt(p: ^longint); begin end;',
  26621. 'begin',
  26622. '']);
  26623. SetExpectedPasResolverError('Not supported: pointer',nNotSupportedX);
  26624. ConvertProgram;
  26625. end;
  26626. procedure TTestModule.TestPointer_AnonymousVarTypeFail;
  26627. begin
  26628. StartProgram(false);
  26629. Add([
  26630. 'var p: ^longint;',
  26631. 'begin',
  26632. '']);
  26633. SetExpectedPasResolverError('Not supported: pointer',nNotSupportedX);
  26634. ConvertProgram;
  26635. end;
  26636. procedure TTestModule.TestPointer_AnonymousResultTypeFail;
  26637. begin
  26638. StartProgram(false);
  26639. Add([
  26640. 'function DoIt: ^longint; begin end;',
  26641. 'begin',
  26642. '']);
  26643. SetExpectedPasResolverError('Not supported: pointer',nNotSupportedX);
  26644. ConvertProgram;
  26645. end;
  26646. procedure TTestModule.TestPointer_AddrOperatorFail;
  26647. begin
  26648. StartProgram(false);
  26649. Add([
  26650. 'var i: longint;',
  26651. 'begin',
  26652. ' if @i=nil then ;',
  26653. '']);
  26654. SetExpectedConverterError('illegal qualifier "@" in front of "i:Longint"',nIllegalQualifierInFrontOf);
  26655. ConvertProgram;
  26656. end;
  26657. procedure TTestModule.TestPointer_ArrayParamsFail;
  26658. begin
  26659. StartProgram(false);
  26660. Add([
  26661. 'var',
  26662. ' p: Pointer;',
  26663. 'begin',
  26664. ' p:=p[1];',
  26665. '']);
  26666. SetExpectedPasResolverError('illegal qualifier "[" after "Pointer"',nIllegalQualifierAfter);
  26667. ConvertProgram;
  26668. end;
  26669. procedure TTestModule.TestPointer_PointerAddFail;
  26670. begin
  26671. StartProgram(false);
  26672. Add([
  26673. 'var',
  26674. ' p: Pointer;',
  26675. 'begin',
  26676. ' p:=p+1;',
  26677. '']);
  26678. SetExpectedPasResolverError('Operator is not overloaded: "Pointer" + "Longint"',nOperatorIsNotOverloadedAOpB);
  26679. ConvertProgram;
  26680. end;
  26681. procedure TTestModule.TestPointer_IncPointerFail;
  26682. begin
  26683. StartProgram(false);
  26684. Add([
  26685. 'var',
  26686. ' p: Pointer;',
  26687. 'begin',
  26688. ' inc(p,1);',
  26689. '']);
  26690. SetExpectedPasResolverError('Incompatible type arg no. 1: Got "Pointer", expected "integer"',
  26691. nIncompatibleTypeArgNo);
  26692. ConvertProgram;
  26693. end;
  26694. procedure TTestModule.TestPointer_Record;
  26695. begin
  26696. StartProgram(false);
  26697. Add([
  26698. 'type',
  26699. ' TRec = record x: longint; end;',
  26700. ' PRec = ^TRec;',
  26701. 'var',
  26702. ' r: TRec;',
  26703. ' p: PRec;',
  26704. ' q: ^TRec;',
  26705. ' Ptr: pointer;',
  26706. 'begin',
  26707. ' new(p);',
  26708. ' p:=@r;',
  26709. ' r:=p^;',
  26710. ' r.x:=p^.x;',
  26711. ' p^.x:=r.x;',
  26712. ' if p^.x=3 then ;',
  26713. ' if 4=p^.x then ;',
  26714. ' dispose(p);',
  26715. ' new(q);',
  26716. ' dispose(q);',
  26717. ' Ptr:=p;',
  26718. ' p:=PRec(ptr);',
  26719. '']);
  26720. ConvertProgram;
  26721. CheckSource('TestPointer_Record',
  26722. LinesToStr([ // statements
  26723. 'rtl.recNewT(this, "TRec", function () {',
  26724. ' this.x = 0;',
  26725. ' this.$eq = function (b) {',
  26726. ' return this.x === b.x;',
  26727. ' };',
  26728. ' this.$assign = function (s) {',
  26729. ' this.x = s.x;',
  26730. ' return this;',
  26731. ' };',
  26732. '});',
  26733. 'this.r = this.TRec.$new();',
  26734. 'this.p = null;',
  26735. 'this.q = null;',
  26736. 'this.Ptr = null;',
  26737. '']),
  26738. LinesToStr([ // $mod.$main
  26739. '$mod.p = $mod.TRec.$new();',
  26740. '$mod.p = $mod.r;',
  26741. '$mod.r.$assign($mod.p);',
  26742. '$mod.r.x = $mod.p.x;',
  26743. '$mod.p.x = $mod.r.x;',
  26744. 'if ($mod.p.x === 3) ;',
  26745. 'if (4 === $mod.p.x) ;',
  26746. '$mod.p = null;',
  26747. '$mod.q = $mod.TRec.$new();',
  26748. '$mod.q = null;',
  26749. '$mod.Ptr = $mod.p;',
  26750. '$mod.p = $mod.Ptr;',
  26751. '']));
  26752. end;
  26753. procedure TTestModule.TestPointer_RecordArg;
  26754. begin
  26755. StartProgram(false);
  26756. Add([
  26757. '{$modeswitch autoderef}',
  26758. 'type',
  26759. ' TRec = record x: longint; end;',
  26760. ' PRec = ^TRec;',
  26761. 'function DoIt(const a: PRec; var b: PRec; out c: PRec): TRec;',
  26762. 'begin',
  26763. ' a.x:=a.x;',
  26764. ' a^.x:=a^.x;',
  26765. ' with a^ do',
  26766. ' x:=x;',
  26767. 'end;',
  26768. 'function GetIt(p: PRec): PRec;',
  26769. 'begin',
  26770. ' p.x:=p.x;',
  26771. ' p^.x:=p^.x;',
  26772. ' with p^ do',
  26773. ' x:=x;',
  26774. 'end;',
  26775. 'var',
  26776. ' r: TRec;',
  26777. ' p: PRec;',
  26778. 'begin',
  26779. ' p:=GetIt(p);',
  26780. ' p^:=GetIt(@r)^;',
  26781. ' DoIt(p,p,p);',
  26782. ' DoIt(@r,p,p);',
  26783. '']);
  26784. ConvertProgram;
  26785. CheckSource('TestPointer_RecordArg',
  26786. LinesToStr([ // statements
  26787. 'rtl.recNewT(this, "TRec", function () {',
  26788. ' this.x = 0;',
  26789. ' this.$eq = function (b) {',
  26790. ' return this.x === b.x;',
  26791. ' };',
  26792. ' this.$assign = function (s) {',
  26793. ' this.x = s.x;',
  26794. ' return this;',
  26795. ' };',
  26796. '});',
  26797. 'this.DoIt = function (a, b, c) {',
  26798. ' var Result = $mod.TRec.$new();',
  26799. ' a.x = a.x;',
  26800. ' a.x = a.x;',
  26801. ' a.x = a.x;',
  26802. ' return Result;',
  26803. '};',
  26804. 'this.GetIt = function (p) {',
  26805. ' var Result = null;',
  26806. ' p.x = p.x;',
  26807. ' p.x = p.x;',
  26808. ' p.x = p.x;',
  26809. ' return Result;',
  26810. '};',
  26811. 'this.r = this.TRec.$new();',
  26812. 'this.p = null;',
  26813. '']),
  26814. LinesToStr([ // $mod.$main
  26815. '$mod.p = $mod.GetIt($mod.p);',
  26816. '$mod.p.$assign($mod.GetIt($mod.r));',
  26817. '$mod.DoIt($mod.p, {',
  26818. ' p: $mod,',
  26819. ' get: function () {',
  26820. ' return this.p.p;',
  26821. ' },',
  26822. ' set: function (v) {',
  26823. ' this.p.p = v;',
  26824. ' }',
  26825. '}, {',
  26826. ' p: $mod,',
  26827. ' get: function () {',
  26828. ' return this.p.p;',
  26829. ' },',
  26830. ' set: function (v) {',
  26831. ' this.p.p = v;',
  26832. ' }',
  26833. '});',
  26834. '$mod.DoIt($mod.r, {',
  26835. ' p: $mod,',
  26836. ' get: function () {',
  26837. ' return this.p.p;',
  26838. ' },',
  26839. ' set: function (v) {',
  26840. ' this.p.p = v;',
  26841. ' }',
  26842. '}, {',
  26843. ' p: $mod,',
  26844. ' get: function () {',
  26845. ' return this.p.p;',
  26846. ' },',
  26847. ' set: function (v) {',
  26848. ' this.p.p = v;',
  26849. ' }',
  26850. '});',
  26851. '']));
  26852. end;
  26853. procedure TTestModule.TestJSValue_AssignToJSValue;
  26854. begin
  26855. StartProgram(false);
  26856. Add('var');
  26857. Add(' v: jsvalue;');
  26858. Add(' i: longint;');
  26859. Add(' s: string;');
  26860. Add(' b: boolean;');
  26861. Add(' d: double;');
  26862. Add(' p: pointer;');
  26863. Add('begin');
  26864. Add(' v:=v;');
  26865. Add(' v:=1;');
  26866. Add(' v:=i;');
  26867. Add(' v:='''';');
  26868. Add(' v:=''c'';');
  26869. Add(' v:=''foo'';');
  26870. Add(' v:=s;');
  26871. Add(' v:=false;');
  26872. Add(' v:=true;');
  26873. Add(' v:=b;');
  26874. Add(' v:=0.1;');
  26875. Add(' v:=d;');
  26876. Add(' v:=nil;');
  26877. Add(' v:=p;');
  26878. ConvertProgram;
  26879. CheckSource('TestJSValue_AssignToJSValue',
  26880. LinesToStr([ // statements
  26881. 'this.v = undefined;',
  26882. 'this.i = 0;',
  26883. 'this.s = "";',
  26884. 'this.b = false;',
  26885. 'this.d = 0.0;',
  26886. 'this.p = null;',
  26887. '']),
  26888. LinesToStr([ // $mod.$main
  26889. '$mod.v = $mod.v;',
  26890. '$mod.v = 1;',
  26891. '$mod.v = $mod.i;',
  26892. '$mod.v = "";',
  26893. '$mod.v = "c";',
  26894. '$mod.v = "foo";',
  26895. '$mod.v = $mod.s;',
  26896. '$mod.v = false;',
  26897. '$mod.v = true;',
  26898. '$mod.v = $mod.b;',
  26899. '$mod.v = 0.1;',
  26900. '$mod.v = $mod.d;',
  26901. '$mod.v = null;',
  26902. '$mod.v = $mod.p;',
  26903. '']));
  26904. end;
  26905. procedure TTestModule.TestJSValue_TypeCastToBaseType;
  26906. begin
  26907. StartProgram(false);
  26908. Add('type');
  26909. Add(' integer = longint;');
  26910. Add(' TYesNo = boolean;');
  26911. Add(' TFloat = double;');
  26912. Add(' TCaption = string;');
  26913. Add(' TChar = char;');
  26914. Add('var');
  26915. Add(' v: jsvalue;');
  26916. Add(' i: integer;');
  26917. Add(' s: TCaption;');
  26918. Add(' b: TYesNo;');
  26919. Add(' d: TFloat;');
  26920. Add(' c: char;');
  26921. Add('begin');
  26922. Add(' i:=longint(v);');
  26923. Add(' i:=integer(v);');
  26924. Add(' s:=string(v);');
  26925. Add(' s:=TCaption(v);');
  26926. Add(' b:=boolean(v);');
  26927. Add(' b:=TYesNo(v);');
  26928. Add(' d:=double(v);');
  26929. Add(' d:=TFloat(v);');
  26930. Add(' c:=char(v);');
  26931. Add(' c:=TChar(v);');
  26932. ConvertProgram;
  26933. CheckSource('TestJSValue_TypeCastToBaseType',
  26934. LinesToStr([ // statements
  26935. 'this.v = undefined;',
  26936. 'this.i = 0;',
  26937. 'this.s = "";',
  26938. 'this.b = false;',
  26939. 'this.d = 0.0;',
  26940. 'this.c = "";',
  26941. '']),
  26942. LinesToStr([ // $mod.$main
  26943. '$mod.i = rtl.trunc($mod.v);',
  26944. '$mod.i = rtl.trunc($mod.v);',
  26945. '$mod.s = "" + $mod.v;',
  26946. '$mod.s = "" + $mod.v;',
  26947. '$mod.b = !($mod.v == false);',
  26948. '$mod.b = !($mod.v == false);',
  26949. '$mod.d = rtl.getNumber($mod.v);',
  26950. '$mod.d = rtl.getNumber($mod.v);',
  26951. '$mod.c = rtl.getChar($mod.v);',
  26952. '$mod.c = rtl.getChar($mod.v);',
  26953. '']));
  26954. end;
  26955. procedure TTestModule.TestJSValue_TypecastToJSValue;
  26956. begin
  26957. StartProgram(false);
  26958. Add([
  26959. 'type',
  26960. ' TArr = array of word;',
  26961. ' TRec = record end;',
  26962. ' TSet = set of boolean;',
  26963. 'procedure Fly(v: jsvalue);',
  26964. 'begin',
  26965. 'end;',
  26966. 'var',
  26967. ' a: TArr;',
  26968. ' r: TRec;',
  26969. ' s: TSet;',
  26970. 'begin',
  26971. ' Fly(jsvalue(a));',
  26972. ' Fly(jsvalue(r));',
  26973. ' Fly(jsvalue(s));',
  26974. '']);
  26975. ConvertProgram;
  26976. CheckSource('TestJSValue_TypecastToJSValue',
  26977. LinesToStr([ // statements
  26978. 'rtl.recNewT(this, "TRec", function () {',
  26979. ' this.$eq = function (b) {',
  26980. ' return true;',
  26981. ' };',
  26982. ' this.$assign = function (s) {',
  26983. ' return this;',
  26984. ' };',
  26985. '});',
  26986. 'this.Fly = function (v) {',
  26987. '};',
  26988. 'this.a = [];',
  26989. 'this.r = this.TRec.$new();',
  26990. 'this.s = {};',
  26991. '']),
  26992. LinesToStr([ // $mod.$main
  26993. '$mod.Fly($mod.a);',
  26994. '$mod.Fly($mod.r);',
  26995. '$mod.Fly($mod.s);',
  26996. '']));
  26997. end;
  26998. procedure TTestModule.TestJSValue_Equal;
  26999. begin
  27000. StartProgram(false);
  27001. Add('type');
  27002. Add(' integer = longint;');
  27003. Add(' TYesNo = boolean;');
  27004. Add(' TFloat = double;');
  27005. Add(' TCaption = string;');
  27006. Add(' TChar = char;');
  27007. Add(' TMulti = JSValue;');
  27008. Add('var');
  27009. Add(' v: jsvalue;');
  27010. Add(' i: integer;');
  27011. Add(' s: TCaption;');
  27012. Add(' b: TYesNo;');
  27013. Add(' d: TFloat;');
  27014. Add(' c: char;');
  27015. Add(' m: TMulti;');
  27016. Add('begin');
  27017. Add(' b:=v=v;');
  27018. Add(' b:=v<>v;');
  27019. Add(' b:=v=1;');
  27020. Add(' b:=v<>1;');
  27021. Add(' b:=2=v;');
  27022. Add(' b:=2<>v;');
  27023. Add(' b:=v=i;');
  27024. Add(' b:=i=v;');
  27025. Add(' b:=v=nil;');
  27026. Add(' b:=nil=v;');
  27027. Add(' b:=v=false;');
  27028. Add(' b:=true=v;');
  27029. Add(' b:=v=b;');
  27030. Add(' b:=b=v;');
  27031. Add(' b:=v=s;');
  27032. Add(' b:=s=v;');
  27033. Add(' b:=v=''foo'';');
  27034. Add(' b:=''''=v;');
  27035. Add(' b:=v=d;');
  27036. Add(' b:=d=v;');
  27037. Add(' b:=v=3.4;');
  27038. Add(' b:=5.6=v;');
  27039. Add(' b:=v=c;');
  27040. Add(' b:=c=v;');
  27041. Add(' b:=m=m;');
  27042. Add(' b:=v=m;');
  27043. Add(' b:=m=v;');
  27044. ConvertProgram;
  27045. CheckSource('TestJSValue_Equal',
  27046. LinesToStr([ // statements
  27047. 'this.v = undefined;',
  27048. 'this.i = 0;',
  27049. 'this.s = "";',
  27050. 'this.b = false;',
  27051. 'this.d = 0.0;',
  27052. 'this.c = "";',
  27053. 'this.m = undefined;',
  27054. '']),
  27055. LinesToStr([ // $mod.$main
  27056. '$mod.b = $mod.v == $mod.v;',
  27057. '$mod.b = $mod.v != $mod.v;',
  27058. '$mod.b = $mod.v == 1;',
  27059. '$mod.b = $mod.v != 1;',
  27060. '$mod.b = 2 == $mod.v;',
  27061. '$mod.b = 2 != $mod.v;',
  27062. '$mod.b = $mod.v == $mod.i;',
  27063. '$mod.b = $mod.i == $mod.v;',
  27064. '$mod.b = $mod.v == null;',
  27065. '$mod.b = null == $mod.v;',
  27066. '$mod.b = $mod.v == false;',
  27067. '$mod.b = true == $mod.v;',
  27068. '$mod.b = $mod.v == $mod.b;',
  27069. '$mod.b = $mod.b == $mod.v;',
  27070. '$mod.b = $mod.v == $mod.s;',
  27071. '$mod.b = $mod.s == $mod.v;',
  27072. '$mod.b = $mod.v == "foo";',
  27073. '$mod.b = "" == $mod.v;',
  27074. '$mod.b = $mod.v == $mod.d;',
  27075. '$mod.b = $mod.d == $mod.v;',
  27076. '$mod.b = $mod.v == 3.4;',
  27077. '$mod.b = 5.6 == $mod.v;',
  27078. '$mod.b = $mod.v == $mod.c;',
  27079. '$mod.b = $mod.c == $mod.v;',
  27080. '$mod.b = $mod.m == $mod.m;',
  27081. '$mod.b = $mod.v == $mod.m;',
  27082. '$mod.b = $mod.m == $mod.v;',
  27083. '']));
  27084. end;
  27085. procedure TTestModule.TestJSValue_If;
  27086. begin
  27087. StartProgram(false);
  27088. Add([
  27089. 'procedure Fly(var u);',
  27090. 'begin',
  27091. ' if jsvalue(u) then ;',
  27092. 'end;',
  27093. 'var',
  27094. ' v: jsvalue;',
  27095. 'begin',
  27096. ' if v then ;',
  27097. ' while v do ;',
  27098. ' repeat until v;',
  27099. '']);
  27100. ConvertProgram;
  27101. CheckSource('TestJSValue_If',
  27102. LinesToStr([ // statements
  27103. 'this.Fly = function (u) {',
  27104. ' if (u.get()) ;',
  27105. '};',
  27106. 'this.v = undefined;',
  27107. '']),
  27108. LinesToStr([ // $mod.$main
  27109. 'if ($mod.v) ;',
  27110. 'while($mod.v){',
  27111. '};',
  27112. 'do{',
  27113. '} while(!$mod.v);',
  27114. '']));
  27115. end;
  27116. procedure TTestModule.TestJSValue_Not;
  27117. begin
  27118. StartProgram(false);
  27119. Add([
  27120. 'var',
  27121. ' v: jsvalue;',
  27122. ' b: boolean;',
  27123. 'begin',
  27124. ' b:=not v;',
  27125. ' if not v then ;',
  27126. ' while not v do ;',
  27127. ' repeat until not v;',
  27128. '']);
  27129. ConvertProgram;
  27130. CheckSource('TestJSValue_If',
  27131. LinesToStr([ // statements
  27132. 'this.v = undefined;',
  27133. 'this.b = false;',
  27134. '']),
  27135. LinesToStr([ // $mod.$main
  27136. '$mod.b=!$mod.v;',
  27137. 'if (!$mod.v) ;',
  27138. 'while(!$mod.v){',
  27139. '};',
  27140. 'do{',
  27141. '} while($mod.v);',
  27142. '']));
  27143. end;
  27144. procedure TTestModule.TestJSValue_Enum;
  27145. begin
  27146. StartProgram(false);
  27147. Add('type');
  27148. Add(' TColor = (red, blue);');
  27149. Add(' TRedBlue = TColor;');
  27150. Add('var');
  27151. Add(' v: jsvalue;');
  27152. Add(' e: TColor;');
  27153. Add('begin');
  27154. Add(' v:=e;');
  27155. Add(' v:=TColor(e);');
  27156. Add(' v:=TRedBlue(e);');
  27157. Add(' e:=TColor(v);');
  27158. Add(' e:=TRedBlue(v);');
  27159. ConvertProgram;
  27160. CheckSource('TestJSValue_Enum',
  27161. LinesToStr([ // statements
  27162. 'this.TColor = {',
  27163. ' "0": "red",',
  27164. ' red: 0,',
  27165. ' "1": "blue",',
  27166. ' blue: 1',
  27167. '};',
  27168. 'this.v = undefined;',
  27169. 'this.e = 0;',
  27170. '']),
  27171. LinesToStr([ // $mod.$main
  27172. '$mod.v = $mod.e;',
  27173. '$mod.v = $mod.e;',
  27174. '$mod.v = $mod.e;',
  27175. '$mod.e = $mod.v;',
  27176. '$mod.e = $mod.v;',
  27177. '']));
  27178. end;
  27179. procedure TTestModule.TestJSValue_ClassInstance;
  27180. begin
  27181. StartProgram(false);
  27182. Add([
  27183. 'type',
  27184. ' TObject = class',
  27185. ' end;',
  27186. ' TBirdObject = TObject;',
  27187. 'var',
  27188. ' v: jsvalue;',
  27189. ' o: TObject;',
  27190. 'begin',
  27191. ' v:=o;',
  27192. ' v:=TObject(o);',
  27193. ' v:=TBirdObject(o);',
  27194. ' o:=TObject(v);',
  27195. ' o:=TBirdObject(v);',
  27196. ' if v is TObject then ;',
  27197. '']);
  27198. ConvertProgram;
  27199. CheckSource('TestJSValue_ClassInstance',
  27200. LinesToStr([ // statements
  27201. 'rtl.createClass(this, "TObject", null, function () {',
  27202. ' this.$init = function () {',
  27203. ' };',
  27204. ' this.$final = function () {',
  27205. ' };',
  27206. '});',
  27207. 'this.v = undefined;',
  27208. 'this.o = null;',
  27209. '']),
  27210. LinesToStr([ // $mod.$main
  27211. '$mod.v = $mod.o;',
  27212. '$mod.v = $mod.o;',
  27213. '$mod.v = $mod.o;',
  27214. '$mod.o = rtl.getObject($mod.v);',
  27215. '$mod.o = rtl.getObject($mod.v);',
  27216. 'if (rtl.isExt($mod.v, $mod.TObject, 1)) ;',
  27217. '']));
  27218. end;
  27219. procedure TTestModule.TestJSValue_ClassOf;
  27220. begin
  27221. StartProgram(false);
  27222. Add([
  27223. 'type',
  27224. ' TClass = class of TObject;',
  27225. ' TObject = class',
  27226. ' end;',
  27227. ' TBirds = class of TBird;',
  27228. ' TBird = class(TObject) end;',
  27229. 'var',
  27230. ' v: jsvalue;',
  27231. ' c: TClass;',
  27232. 'begin',
  27233. ' v:=c;',
  27234. ' v:=TObject;',
  27235. ' v:=TClass(c);',
  27236. ' v:=TBirds(c);',
  27237. ' c:=TClass(v);',
  27238. ' c:=TBirds(v);',
  27239. ' if v is TClass then ;',
  27240. '']);
  27241. ConvertProgram;
  27242. CheckSource('TestJSValue_ClassOf',
  27243. LinesToStr([ // statements
  27244. 'rtl.createClass(this, "TObject", null, function () {',
  27245. ' this.$init = function () {',
  27246. ' };',
  27247. ' this.$final = function () {',
  27248. ' };',
  27249. '});',
  27250. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  27251. '});',
  27252. 'this.v = undefined;',
  27253. 'this.c = null;',
  27254. '']),
  27255. LinesToStr([ // $mod.$main
  27256. '$mod.v = $mod.c;',
  27257. '$mod.v = $mod.TObject;',
  27258. '$mod.v = $mod.c;',
  27259. '$mod.v = $mod.c;',
  27260. '$mod.c = rtl.getObject($mod.v);',
  27261. '$mod.c = rtl.getObject($mod.v);',
  27262. 'if (rtl.isExt($mod.v, $mod.TObject, 2)) ;',
  27263. '']));
  27264. end;
  27265. procedure TTestModule.TestJSValue_ArrayOfJSValue;
  27266. begin
  27267. StartProgram(false);
  27268. Add([
  27269. 'type',
  27270. ' integer = longint;',
  27271. ' TArray = array of JSValue;',
  27272. ' TArrgh = tarray;',
  27273. ' TArrInt = array of integer;',
  27274. 'var',
  27275. ' v: jsvalue;',
  27276. ' TheArray: tarray = (1,''2'');',
  27277. ' Arr: tarrgh;',
  27278. ' i: integer;',
  27279. ' ArrInt: tarrint;',
  27280. 'begin',
  27281. ' arr:=thearray;',
  27282. ' thearray:=arr;',
  27283. ' setlength(arr,2);',
  27284. ' setlength(thearray,3);',
  27285. ' arr[4]:=v;',
  27286. ' arr[5]:=length(thearray);',
  27287. ' arr[6]:=nil;',
  27288. ' arr[7]:=thearray[8];',
  27289. ' arr[low(arr)]:=high(thearray);',
  27290. ' arr:=arrint;',
  27291. ' arrInt:=tarrint(arr);',
  27292. ' if TheArray = nil then ;',
  27293. ' if nil = TheArray then ;',
  27294. ' if TheArray <> nil then ;',
  27295. ' if nil <> TheArray then ;',
  27296. '']);
  27297. ConvertProgram;
  27298. CheckSource('TestJSValue_ArrayOfJSValue',
  27299. LinesToStr([ // statements
  27300. 'this.v = undefined;',
  27301. 'this.TheArray = [1, "2"];',
  27302. 'this.Arr = [];',
  27303. 'this.i = 0;',
  27304. 'this.ArrInt = [];',
  27305. '']),
  27306. LinesToStr([ // $mod.$main
  27307. '$mod.Arr = rtl.arrayRef($mod.TheArray);',
  27308. '$mod.TheArray = rtl.arrayRef($mod.Arr);',
  27309. '$mod.Arr = rtl.arraySetLength($mod.Arr,undefined,2);',
  27310. '$mod.TheArray = rtl.arraySetLength($mod.TheArray,undefined,3);',
  27311. '$mod.Arr[4] = $mod.v;',
  27312. '$mod.Arr[5] = rtl.length($mod.TheArray);',
  27313. '$mod.Arr[6] = null;',
  27314. '$mod.Arr[7] = $mod.TheArray[8];',
  27315. '$mod.Arr[0] = rtl.length($mod.TheArray) - 1;',
  27316. '$mod.Arr = rtl.arrayRef($mod.ArrInt);',
  27317. '$mod.ArrInt = $mod.Arr;',
  27318. 'if (rtl.length($mod.TheArray) === 0) ;',
  27319. 'if (rtl.length($mod.TheArray) === 0) ;',
  27320. 'if (rtl.length($mod.TheArray) > 0) ;',
  27321. 'if (rtl.length($mod.TheArray) > 0) ;',
  27322. '']));
  27323. end;
  27324. procedure TTestModule.TestJSValue_ArrayLit;
  27325. begin
  27326. StartProgram(false);
  27327. Add([
  27328. 'type',
  27329. ' TFlag = (big,small);',
  27330. ' TArray = array of JSValue;',
  27331. ' TObject = class end;',
  27332. ' TClass = class of TObject;',
  27333. 'var',
  27334. ' v: jsvalue;',
  27335. ' a: TArray;',
  27336. ' o: TObject;',
  27337. 'begin',
  27338. ' a:=[];',
  27339. ' a:=[1];',
  27340. ' a:=[1,2];',
  27341. ' a:=[big];',
  27342. ' a:=[1,big];',
  27343. ' a:=[o,nil];',
  27344. '']);
  27345. ConvertProgram;
  27346. CheckSource('TestJSValue_ArrayLit',
  27347. LinesToStr([ // statements
  27348. 'this.TFlag = {',
  27349. ' "0": "big",',
  27350. ' big: 0,',
  27351. ' "1": "small",',
  27352. ' small: 1',
  27353. '};',
  27354. 'rtl.createClass(this, "TObject", null, function () {',
  27355. ' this.$init = function () {',
  27356. ' };',
  27357. ' this.$final = function () {',
  27358. ' };',
  27359. '});',
  27360. 'this.v = undefined;',
  27361. 'this.a = [];',
  27362. 'this.o = null;',
  27363. '']),
  27364. LinesToStr([ // $mod.$main
  27365. '$mod.a = [];',
  27366. '$mod.a = [1];',
  27367. '$mod.a = [1, 2];',
  27368. '$mod.a = [$mod.TFlag.big];',
  27369. '$mod.a = [1, $mod.TFlag.big];',
  27370. '$mod.a = [$mod.o, null];',
  27371. '']));
  27372. end;
  27373. procedure TTestModule.TestJSValue_Params;
  27374. begin
  27375. StartProgram(false);
  27376. Add('type');
  27377. Add(' integer = longint;');
  27378. Add(' TYesNo = boolean;');
  27379. Add(' TFloat = double;');
  27380. Add(' TCaption = string;');
  27381. Add(' TChar = char;');
  27382. Add('function DoIt(a: jsvalue; const b: jsvalue; var c: jsvalue; out d: jsvalue): jsvalue;');
  27383. Add('var');
  27384. Add(' l: jsvalue;');
  27385. Add('begin');
  27386. Add(' a:=a;');
  27387. Add(' l:=b;');
  27388. Add(' c:=c;');
  27389. Add(' d:=d;');
  27390. Add(' Result:=l;');
  27391. Add('end;');
  27392. Add('function DoSome(a: jsvalue; const b: jsvalue): jsvalue; begin end;');
  27393. Add('var');
  27394. Add(' v: jsvalue;');
  27395. Add(' i: integer;');
  27396. Add(' b: TYesNo;');
  27397. Add(' d: TFloat;');
  27398. Add(' s: TCaption;');
  27399. Add(' c: TChar;');
  27400. Add('begin');
  27401. Add(' v:=doit(v,v,v,v);');
  27402. Add(' i:=integer(dosome(i,i));');
  27403. Add(' b:=TYesNo(dosome(b,b));');
  27404. Add(' d:=TFloat(dosome(d,d));');
  27405. Add(' s:=TCaption(dosome(s,s));');
  27406. Add(' c:=TChar(dosome(c,c));');
  27407. ConvertProgram;
  27408. CheckSource('TestJSValue_Params',
  27409. LinesToStr([ // statements
  27410. 'this.DoIt = function (a, b, c, d) {',
  27411. ' var Result = undefined;',
  27412. ' var l = undefined;',
  27413. ' a = a;',
  27414. ' l = b;',
  27415. ' c.set(c.get());',
  27416. ' d.set(d.get());',
  27417. ' Result = l;',
  27418. ' return Result;',
  27419. '};',
  27420. 'this.DoSome = function (a, b) {',
  27421. ' var Result = undefined;',
  27422. ' return Result;',
  27423. '};',
  27424. 'this.v = undefined;',
  27425. 'this.i = 0;',
  27426. 'this.b = false;',
  27427. 'this.d = 0.0;',
  27428. 'this.s = "";',
  27429. 'this.c = "";',
  27430. '']),
  27431. LinesToStr([ // $mod.$main
  27432. '$mod.v = $mod.DoIt($mod.v, $mod.v, {',
  27433. ' p: $mod,',
  27434. ' get: function () {',
  27435. ' return this.p.v;',
  27436. ' },',
  27437. ' set: function (v) {',
  27438. ' this.p.v = v;',
  27439. ' }',
  27440. '}, {',
  27441. ' p: $mod,',
  27442. ' get: function () {',
  27443. ' return this.p.v;',
  27444. ' },',
  27445. ' set: function (v) {',
  27446. ' this.p.v = v;',
  27447. ' }',
  27448. '});',
  27449. '$mod.i = rtl.trunc($mod.DoSome($mod.i, $mod.i));',
  27450. '$mod.b = !($mod.DoSome($mod.b, $mod.b) == false);',
  27451. '$mod.d = rtl.getNumber($mod.DoSome($mod.d, $mod.d));',
  27452. '$mod.s = "" + $mod.DoSome($mod.s, $mod.s);',
  27453. '$mod.c = rtl.getChar($mod.DoSome($mod.c, $mod.c));',
  27454. '']));
  27455. end;
  27456. procedure TTestModule.TestJSValue_UntypedParam;
  27457. begin
  27458. StartProgram(false);
  27459. Add('function DoIt(const a; var b; out c): jsvalue;');
  27460. Add('begin');
  27461. Add(' Result:=a;');
  27462. Add(' Result:=b;');
  27463. Add(' Result:=c;');
  27464. Add(' b:=Result;');
  27465. Add(' c:=Result;');
  27466. Add('end;');
  27467. Add('var i: longint;');
  27468. Add('begin');
  27469. Add(' doit(i,i,i);');
  27470. ConvertProgram;
  27471. CheckSource('TestJSValue_UntypedParam',
  27472. LinesToStr([ // statements
  27473. 'this.DoIt = function (a, b, c) {',
  27474. ' var Result = undefined;',
  27475. ' Result = a;',
  27476. ' Result = b.get();',
  27477. ' Result = c.get();',
  27478. ' b.set(Result);',
  27479. ' c.set(Result);',
  27480. ' return Result;',
  27481. '};',
  27482. 'this.i = 0;',
  27483. '']),
  27484. LinesToStr([ // $mod.$main
  27485. '$mod.DoIt($mod.i, {',
  27486. ' p: $mod,',
  27487. ' get: function () {',
  27488. ' return this.p.i;',
  27489. ' },',
  27490. ' set: function (v) {',
  27491. ' this.p.i = v;',
  27492. ' }',
  27493. '}, {',
  27494. ' p: $mod,',
  27495. ' get: function () {',
  27496. ' return this.p.i;',
  27497. ' },',
  27498. ' set: function (v) {',
  27499. ' this.p.i = v;',
  27500. ' }',
  27501. '});',
  27502. '']));
  27503. end;
  27504. procedure TTestModule.TestJSValue_FuncResultType;
  27505. begin
  27506. StartProgram(false);
  27507. Add('type');
  27508. Add(' integer = longint;');
  27509. Add(' TJSValueArray = array of JSValue;');
  27510. Add(' TListSortCompare = function(Item1, Item2: JSValue): Integer;');
  27511. Add('procedure Sort(P: JSValue; aList: TJSValueArray; const Compare: TListSortCompare);');
  27512. Add('begin');
  27513. Add(' while Compare(P,aList[0])>0 do ;');
  27514. Add('end;');
  27515. Add('var');
  27516. Add(' Compare: TListSortCompare;');
  27517. Add(' V: JSValue;');
  27518. Add(' i: integer;');
  27519. Add('begin');
  27520. Add(' if Compare(V,V)>0 then ;');
  27521. Add(' if Compare(i,i)>1 then ;');
  27522. Add(' if Compare(nil,false)>2 then ;');
  27523. Add(' if Compare(1,true)>3 then ;');
  27524. ConvertProgram;
  27525. CheckSource('TestJSValue_UntypedParam',
  27526. LinesToStr([ // statements
  27527. 'this.Sort = function (P, aList, Compare) {',
  27528. ' while (Compare(P, aList[0]) > 0) {',
  27529. ' };',
  27530. '};',
  27531. 'this.Compare = null;',
  27532. 'this.V = undefined;',
  27533. 'this.i = 0;',
  27534. '']),
  27535. LinesToStr([ // $mod.$main
  27536. 'if ($mod.Compare($mod.V, $mod.V) > 0) ;',
  27537. 'if ($mod.Compare($mod.i, $mod.i) > 1) ;',
  27538. 'if ($mod.Compare(null, false) > 2) ;',
  27539. 'if ($mod.Compare(1, true) > 3) ;',
  27540. '']));
  27541. end;
  27542. procedure TTestModule.TestJSValue_ProcType_Assign;
  27543. begin
  27544. StartProgram(false);
  27545. Add('type');
  27546. Add(' integer = longint;');
  27547. Add(' TObject = class');
  27548. Add(' class function GetGlob: integer;');
  27549. Add(' function Getter: integer;');
  27550. Add(' end;');
  27551. Add('class function TObject.GetGlob: integer;');
  27552. Add('var v1: jsvalue;');
  27553. Add('begin');
  27554. Add(' v1:=@GetGlob;');
  27555. Add(' v1:[email protected];');
  27556. Add('end;');
  27557. Add('function TObject.Getter: integer;');
  27558. Add('var v2: jsvalue;');
  27559. Add('begin');
  27560. Add(' v2:=@Getter;');
  27561. Add(' v2:[email protected];');
  27562. Add(' v2:=@GetGlob;');
  27563. Add(' v2:[email protected];');
  27564. Add('end;');
  27565. Add('function GetIt(i: integer): integer;');
  27566. Add('var v3: jsvalue;');
  27567. Add('begin');
  27568. Add(' v3:=@GetIt;');
  27569. Add('end;');
  27570. Add('var');
  27571. Add(' V: JSValue;');
  27572. Add(' o: TObject;');
  27573. Add('begin');
  27574. Add(' v:=@GetIt;');
  27575. Add(' v:[email protected];');
  27576. Add(' v:[email protected];');
  27577. ConvertProgram;
  27578. CheckSource('TestJSValue_ProcType_Assign',
  27579. LinesToStr([ // statements
  27580. 'rtl.createClass(this, "TObject", null, function () {',
  27581. ' this.$init = function () {',
  27582. ' };',
  27583. ' this.$final = function () {',
  27584. ' };',
  27585. ' this.GetGlob = function () {',
  27586. ' var Result = 0;',
  27587. ' var v1 = undefined;',
  27588. ' v1 = rtl.createCallback(this, "GetGlob");',
  27589. ' v1 = rtl.createCallback(this, "GetGlob");',
  27590. ' return Result;',
  27591. ' };',
  27592. ' this.Getter = function () {',
  27593. ' var Result = 0;',
  27594. ' var v2 = undefined;',
  27595. ' v2 = rtl.createCallback(this, "Getter");',
  27596. ' v2 = rtl.createCallback(this, "Getter");',
  27597. ' v2 = rtl.createCallback(this.$class, "GetGlob");',
  27598. ' v2 = rtl.createCallback(this.$class, "GetGlob");',
  27599. ' return Result;',
  27600. ' };',
  27601. '});',
  27602. 'this.GetIt = function (i) {',
  27603. ' var Result = 0;',
  27604. ' var v3 = undefined;',
  27605. ' v3 = $mod.GetIt;',
  27606. ' return Result;',
  27607. '};',
  27608. 'this.V = undefined;',
  27609. 'this.o = null;',
  27610. '']),
  27611. LinesToStr([ // $mod.$main
  27612. '$mod.V = $mod.GetIt;',
  27613. '$mod.V = rtl.createCallback($mod.o, "Getter");',
  27614. '$mod.V = rtl.createCallback($mod.o.$class, "GetGlob");',
  27615. '']));
  27616. end;
  27617. procedure TTestModule.TestJSValue_ProcType_Equal;
  27618. begin
  27619. StartProgram(false);
  27620. Add('type');
  27621. Add(' integer = longint;');
  27622. Add(' TObject = class');
  27623. Add(' class function GetGlob: integer;');
  27624. Add(' function Getter: integer;');
  27625. Add(' end;');
  27626. Add('class function TObject.GetGlob: integer;');
  27627. Add('var v1: jsvalue;');
  27628. Add('begin');
  27629. Add(' if v1=@GetGlob then;');
  27630. Add(' if [email protected] then ;');
  27631. Add('end;');
  27632. Add('function TObject.Getter: integer;');
  27633. Add('var v2: jsvalue;');
  27634. Add('begin');
  27635. Add(' if v2=@Getter then;');
  27636. Add(' if [email protected] then ;');
  27637. Add(' if v2=@GetGlob then;');
  27638. Add(' if [email protected] then;');
  27639. Add('end;');
  27640. Add('function GetIt(i: integer): integer;');
  27641. Add('var v3: jsvalue;');
  27642. Add('begin');
  27643. Add(' if v3=@GetIt then;');
  27644. Add('end;');
  27645. Add('var');
  27646. Add(' V: JSValue;');
  27647. Add(' o: TObject;');
  27648. Add('begin');
  27649. Add(' if v=@GetIt then;');
  27650. Add(' if [email protected] then;');
  27651. Add(' if [email protected] then;');
  27652. Add(' if @GetIt=v then;');
  27653. Add(' if @o.Getter=v then;');
  27654. Add(' if @o.GetGlob=v then;');
  27655. ConvertProgram;
  27656. CheckSource('TestJSValue_ProcType_Equal',
  27657. LinesToStr([ // statements
  27658. 'rtl.createClass(this, "TObject", null, function () {',
  27659. ' this.$init = function () {',
  27660. ' };',
  27661. ' this.$final = function () {',
  27662. ' };',
  27663. ' this.GetGlob = function () {',
  27664. ' var Result = 0;',
  27665. ' var v1 = undefined;',
  27666. ' if (rtl.eqCallback(v1, rtl.createCallback(this, "GetGlob"))) ;',
  27667. ' if (rtl.eqCallback(v1, rtl.createCallback(this, "GetGlob"))) ;',
  27668. ' return Result;',
  27669. ' };',
  27670. ' this.Getter = function () {',
  27671. ' var Result = 0;',
  27672. ' var v2 = undefined;',
  27673. ' if (rtl.eqCallback(v2, rtl.createCallback(this, "Getter"))) ;',
  27674. ' if (rtl.eqCallback(v2, rtl.createCallback(this, "Getter"))) ;',
  27675. ' if (rtl.eqCallback(v2, rtl.createCallback(this.$class, "GetGlob"))) ;',
  27676. ' if (rtl.eqCallback(v2, rtl.createCallback(this.$class, "GetGlob"))) ;',
  27677. ' return Result;',
  27678. ' };',
  27679. '});',
  27680. 'this.GetIt = function (i) {',
  27681. ' var Result = 0;',
  27682. ' var v3 = undefined;',
  27683. ' if (rtl.eqCallback(v3, $mod.GetIt)) ;',
  27684. ' return Result;',
  27685. '};',
  27686. 'this.V = undefined;',
  27687. 'this.o = null;',
  27688. '']),
  27689. LinesToStr([ // $mod.$main
  27690. 'if (rtl.eqCallback($mod.V, $mod.GetIt)) ;',
  27691. 'if (rtl.eqCallback($mod.V, rtl.createCallback($mod.o, "Getter"))) ;',
  27692. 'if (rtl.eqCallback($mod.V, rtl.createCallback($mod.o.$class, "GetGlob"))) ;',
  27693. 'if (rtl.eqCallback($mod.GetIt, $mod.V)) ;',
  27694. 'if (rtl.eqCallback(rtl.createCallback($mod.o, "Getter"), $mod.V)) ;',
  27695. 'if (rtl.eqCallback(rtl.createCallback($mod.o.$class, "GetGlob"), $mod.V)) ;',
  27696. '']));
  27697. end;
  27698. procedure TTestModule.TestJSValue_ProcType_Param;
  27699. begin
  27700. StartProgram(false);
  27701. Add([
  27702. 'type',
  27703. ' variant = jsvalue;',
  27704. ' TArrVariant = array of variant;',
  27705. ' TArrVar2 = TArrVariant;',
  27706. ' TFuncInt = function: longint;',
  27707. 'function GetIt: longint;',
  27708. 'begin',
  27709. 'end;',
  27710. 'procedure DoIt(p: jsvalue; Arr: TArrVar2);',
  27711. 'var v: variant;',
  27712. 'begin',
  27713. ' v:=arr[1];',
  27714. 'end;',
  27715. 'var s: string;',
  27716. 'begin',
  27717. ' DoIt(GetIt,[]);',
  27718. ' DoIt(@GetIt,[]);',
  27719. ' DoIt(1,[s,GetIt]);',
  27720. ' DoIt(1,[s,@GetIt]);',
  27721. '']);
  27722. ConvertProgram;
  27723. CheckSource('TestJSValue_ProcType_Param',
  27724. LinesToStr([ // statements
  27725. 'this.GetIt = function () {',
  27726. ' var Result = 0;',
  27727. ' return Result;',
  27728. '};',
  27729. 'this.DoIt = function (p, Arr) {',
  27730. ' var v = undefined;',
  27731. ' v = Arr[1];',
  27732. '};',
  27733. 'this.s = "";',
  27734. '']),
  27735. LinesToStr([ // $mod.$main
  27736. '$mod.DoIt($mod.GetIt(), []);',
  27737. '$mod.DoIt($mod.GetIt, []);',
  27738. '$mod.DoIt(1, [$mod.s, $mod.GetIt()]);',
  27739. '$mod.DoIt(1, [$mod.s, $mod.GetIt]);',
  27740. '']));
  27741. end;
  27742. procedure TTestModule.TestJSValue_AssignToPointerFail;
  27743. begin
  27744. StartProgram(false);
  27745. Add([
  27746. 'var',
  27747. ' v: JSValue;',
  27748. ' p: Pointer;',
  27749. 'begin',
  27750. ' p:=v;',
  27751. '']);
  27752. SetExpectedPasResolverError('Incompatible types: got "JSValue" expected "Pointer"',
  27753. nIncompatibleTypesGotExpected);
  27754. ConvertProgram;
  27755. end;
  27756. procedure TTestModule.TestJSValue_OverloadDouble;
  27757. begin
  27758. StartProgram(false);
  27759. Add([
  27760. 'type',
  27761. ' integer = longint;',
  27762. ' tdatetime = double;',
  27763. 'procedure DoIt(d: double); begin end;',
  27764. 'procedure DoIt(v: jsvalue); begin end;',
  27765. 'var',
  27766. ' d: double;',
  27767. ' dt: tdatetime;',
  27768. ' i: integer;',
  27769. ' b: byte;',
  27770. ' shi: shortint;',
  27771. ' w: word;',
  27772. ' smi: smallint;',
  27773. ' lw: longword;',
  27774. ' li: longint;',
  27775. ' ni: nativeint;',
  27776. ' nu: nativeuint;',
  27777. 'begin',
  27778. ' DoIt(d);',
  27779. ' DoIt(dt);',
  27780. ' DoIt(i);',
  27781. ' DoIt(b);',
  27782. ' DoIt(shi);',
  27783. ' DoIt(w);',
  27784. ' DoIt(smi);',
  27785. ' DoIt(lw);',
  27786. ' DoIt(li);',
  27787. ' DoIt(ni);',
  27788. ' DoIt(nu);',
  27789. '']);
  27790. ConvertProgram;
  27791. CheckSource('TestJSValue_OverloadDouble',
  27792. LinesToStr([ // statements
  27793. 'this.DoIt = function (d) {',
  27794. '};',
  27795. 'this.DoIt$1 = function (v) {',
  27796. '};',
  27797. 'this.d = 0.0;',
  27798. 'this.dt = 0.0;',
  27799. 'this.i = 0;',
  27800. 'this.b = 0;',
  27801. 'this.shi = 0;',
  27802. 'this.w = 0;',
  27803. 'this.smi = 0;',
  27804. 'this.lw = 0;',
  27805. 'this.li = 0;',
  27806. 'this.ni = 0;',
  27807. 'this.nu = 0;',
  27808. '']),
  27809. LinesToStr([ // $mod.$main
  27810. '$mod.DoIt($mod.d);',
  27811. '$mod.DoIt($mod.dt);',
  27812. '$mod.DoIt$1($mod.i);',
  27813. '$mod.DoIt$1($mod.b);',
  27814. '$mod.DoIt$1($mod.shi);',
  27815. '$mod.DoIt$1($mod.w);',
  27816. '$mod.DoIt$1($mod.smi);',
  27817. '$mod.DoIt$1($mod.lw);',
  27818. '$mod.DoIt$1($mod.li);',
  27819. '$mod.DoIt$1($mod.ni);',
  27820. '$mod.DoIt$1($mod.nu);',
  27821. '']));
  27822. end;
  27823. procedure TTestModule.TestJSValue_OverloadNativeInt;
  27824. begin
  27825. StartProgram(false);
  27826. Add([
  27827. 'type',
  27828. ' integer = longint;',
  27829. ' int53 = nativeint;',
  27830. ' tdatetime = double;',
  27831. 'procedure DoIt(n: nativeint); begin end;',
  27832. 'procedure DoIt(v: jsvalue); begin end;',
  27833. 'var',
  27834. ' d: double;',
  27835. ' dt: tdatetime;',
  27836. ' i: integer;',
  27837. ' b: byte;',
  27838. ' shi: shortint;',
  27839. ' w: word;',
  27840. ' smi: smallint;',
  27841. ' lw: longword;',
  27842. ' li: longint;',
  27843. ' ni: nativeint;',
  27844. ' nu: nativeuint;',
  27845. 'begin',
  27846. ' DoIt(d);',
  27847. ' DoIt(dt);',
  27848. ' DoIt(i);',
  27849. ' DoIt(b);',
  27850. ' DoIt(shi);',
  27851. ' DoIt(w);',
  27852. ' DoIt(smi);',
  27853. ' DoIt(lw);',
  27854. ' DoIt(li);',
  27855. ' DoIt(ni);',
  27856. ' DoIt(nu);',
  27857. '']);
  27858. ConvertProgram;
  27859. CheckSource('TestJSValue_OverloadNativeInt',
  27860. LinesToStr([ // statements
  27861. 'this.DoIt = function (n) {',
  27862. '};',
  27863. 'this.DoIt$1 = function (v) {',
  27864. '};',
  27865. 'this.d = 0.0;',
  27866. 'this.dt = 0.0;',
  27867. 'this.i = 0;',
  27868. 'this.b = 0;',
  27869. 'this.shi = 0;',
  27870. 'this.w = 0;',
  27871. 'this.smi = 0;',
  27872. 'this.lw = 0;',
  27873. 'this.li = 0;',
  27874. 'this.ni = 0;',
  27875. 'this.nu = 0;',
  27876. '']),
  27877. LinesToStr([ // $mod.$main
  27878. '$mod.DoIt$1($mod.d);',
  27879. '$mod.DoIt$1($mod.dt);',
  27880. '$mod.DoIt($mod.i);',
  27881. '$mod.DoIt($mod.b);',
  27882. '$mod.DoIt($mod.shi);',
  27883. '$mod.DoIt($mod.w);',
  27884. '$mod.DoIt($mod.smi);',
  27885. '$mod.DoIt($mod.lw);',
  27886. '$mod.DoIt($mod.li);',
  27887. '$mod.DoIt($mod.ni);',
  27888. '$mod.DoIt($mod.nu);',
  27889. '']));
  27890. end;
  27891. procedure TTestModule.TestJSValue_OverloadWord;
  27892. begin
  27893. StartProgram(false);
  27894. Add([
  27895. 'type',
  27896. ' integer = longint;',
  27897. ' int53 = nativeint;',
  27898. ' tdatetime = double;',
  27899. 'procedure DoIt(w: word); begin end;',
  27900. 'procedure DoIt(v: jsvalue); begin end;',
  27901. 'var',
  27902. ' d: double;',
  27903. ' dt: tdatetime;',
  27904. ' i: integer;',
  27905. ' b: byte;',
  27906. ' shi: shortint;',
  27907. ' w: word;',
  27908. ' smi: smallint;',
  27909. ' lw: longword;',
  27910. ' li: longint;',
  27911. ' ni: nativeint;',
  27912. ' nu: nativeuint;',
  27913. 'begin',
  27914. ' DoIt(d);',
  27915. ' DoIt(dt);',
  27916. ' DoIt(i);',
  27917. ' DoIt(b);',
  27918. ' DoIt(shi);',
  27919. ' DoIt(w);',
  27920. ' DoIt(smi);',
  27921. ' DoIt(lw);',
  27922. ' DoIt(li);',
  27923. ' DoIt(ni);',
  27924. ' DoIt(nu);',
  27925. '']);
  27926. ConvertProgram;
  27927. CheckSource('TestJSValue_OverloadWord',
  27928. LinesToStr([ // statements
  27929. 'this.DoIt = function (w) {',
  27930. '};',
  27931. 'this.DoIt$1 = function (v) {',
  27932. '};',
  27933. 'this.d = 0.0;',
  27934. 'this.dt = 0.0;',
  27935. 'this.i = 0;',
  27936. 'this.b = 0;',
  27937. 'this.shi = 0;',
  27938. 'this.w = 0;',
  27939. 'this.smi = 0;',
  27940. 'this.lw = 0;',
  27941. 'this.li = 0;',
  27942. 'this.ni = 0;',
  27943. 'this.nu = 0;',
  27944. '']),
  27945. LinesToStr([ // $mod.$main
  27946. '$mod.DoIt$1($mod.d);',
  27947. '$mod.DoIt$1($mod.dt);',
  27948. '$mod.DoIt$1($mod.i);',
  27949. '$mod.DoIt($mod.b);',
  27950. '$mod.DoIt($mod.shi);',
  27951. '$mod.DoIt($mod.w);',
  27952. '$mod.DoIt$1($mod.smi);',
  27953. '$mod.DoIt$1($mod.lw);',
  27954. '$mod.DoIt$1($mod.li);',
  27955. '$mod.DoIt$1($mod.ni);',
  27956. '$mod.DoIt$1($mod.nu);',
  27957. '']));
  27958. end;
  27959. procedure TTestModule.TestJSValue_OverloadString;
  27960. begin
  27961. StartProgram(false);
  27962. Add([
  27963. 'type',
  27964. ' uni = string;',
  27965. ' WChar = char;',
  27966. 'procedure DoIt(s: string); begin end;',
  27967. 'procedure DoIt(v: jsvalue); begin end;',
  27968. 'var',
  27969. ' s: string;',
  27970. ' c: char;',
  27971. ' u: uni;',
  27972. 'begin',
  27973. ' DoIt(s);',
  27974. ' DoIt(c);',
  27975. ' DoIt(u);',
  27976. '']);
  27977. ConvertProgram;
  27978. CheckSource('TestJSValue_OverloadString',
  27979. LinesToStr([ // statements
  27980. 'this.DoIt = function (s) {',
  27981. '};',
  27982. 'this.DoIt$1 = function (v) {',
  27983. '};',
  27984. 'this.s = "";',
  27985. 'this.c = "";',
  27986. 'this.u = "";',
  27987. '']),
  27988. LinesToStr([ // $mod.$main
  27989. '$mod.DoIt($mod.s);',
  27990. '$mod.DoIt($mod.c);',
  27991. '$mod.DoIt($mod.u);',
  27992. '']));
  27993. end;
  27994. procedure TTestModule.TestJSValue_OverloadChar;
  27995. begin
  27996. StartProgram(false);
  27997. Add([
  27998. 'type',
  27999. ' uni = string;',
  28000. ' WChar = char;',
  28001. 'procedure DoIt(c: char); begin end;',
  28002. 'procedure DoIt(v: jsvalue); begin end;',
  28003. 'var',
  28004. ' s: string;',
  28005. ' c: char;',
  28006. ' u: uni;',
  28007. 'begin',
  28008. ' DoIt(s);',
  28009. ' DoIt(c);',
  28010. ' DoIt(u);',
  28011. '']);
  28012. ConvertProgram;
  28013. CheckSource('TestJSValue_OverloadChar',
  28014. LinesToStr([ // statements
  28015. 'this.DoIt = function (c) {',
  28016. '};',
  28017. 'this.DoIt$1 = function (v) {',
  28018. '};',
  28019. 'this.s = "";',
  28020. 'this.c = "";',
  28021. 'this.u = "";',
  28022. '']),
  28023. LinesToStr([ // $mod.$main
  28024. '$mod.DoIt$1($mod.s);',
  28025. '$mod.DoIt($mod.c);',
  28026. '$mod.DoIt$1($mod.u);',
  28027. '']));
  28028. end;
  28029. procedure TTestModule.TestJSValue_OverloadPointer;
  28030. begin
  28031. StartProgram(false);
  28032. Add([
  28033. 'type',
  28034. ' TObject = class end;',
  28035. 'procedure DoIt(p: pointer); begin end;',
  28036. 'procedure DoIt(v: jsvalue); begin end;',
  28037. 'var',
  28038. ' o: TObject;',
  28039. 'begin',
  28040. ' DoIt(o);',
  28041. '']);
  28042. ConvertProgram;
  28043. CheckSource('TestJSValue_OverloadPointer',
  28044. LinesToStr([ // statements
  28045. 'rtl.createClass(this, "TObject", null, function () {',
  28046. ' this.$init = function () {',
  28047. ' };',
  28048. ' this.$final = function () {',
  28049. ' };',
  28050. '});',
  28051. 'this.DoIt = function (p) {',
  28052. '};',
  28053. 'this.DoIt$1 = function (v) {',
  28054. '};',
  28055. 'this.o = null;',
  28056. '']),
  28057. LinesToStr([ // $mod.$main
  28058. '$mod.DoIt($mod.o);',
  28059. '']));
  28060. end;
  28061. procedure TTestModule.TestJSValue_ForIn;
  28062. begin
  28063. StartProgram(false);
  28064. Add([
  28065. 'var',
  28066. ' v: JSValue;',
  28067. ' key: string;',
  28068. 'begin',
  28069. ' for key in v do begin',
  28070. ' if key=''abc'' then ;',
  28071. ' end;',
  28072. '']);
  28073. ConvertProgram;
  28074. CheckSource('TestJSValue_ForIn',
  28075. LinesToStr([ // statements
  28076. 'this.v = undefined;',
  28077. 'this.key = "";',
  28078. '']),
  28079. LinesToStr([ // $mod.$main
  28080. 'for ($mod.key in $mod.v) {',
  28081. ' if ($mod.key === "abc") ;',
  28082. '};',
  28083. '']));
  28084. end;
  28085. procedure TTestModule.TestRTTI_IntRange;
  28086. begin
  28087. WithTypeInfo:=true;
  28088. StartProgram(true,[supTypeInfo]);
  28089. Add([
  28090. '{$modeswitch externalclass}',
  28091. 'type',
  28092. ' TGraphicsColor = -$7FFFFFFF-1..$7FFFFFFF;',
  28093. ' TColor = type TGraphicsColor;',
  28094. 'var',
  28095. ' p: TTypeInfo;',
  28096. ' k: TTypeKind;',
  28097. 'begin',
  28098. ' p:=typeinfo(TGraphicsColor);',
  28099. ' p:=typeinfo(TColor);',
  28100. ' k:=GetTypeKind(TGraphicsColor);',
  28101. ' k:=GetTypeKind(TColor);',
  28102. '']);
  28103. ConvertProgram;
  28104. CheckSource('TestRTTI_IntRange',
  28105. LinesToStr([ // statements
  28106. 'this.$rtti.$Int("TGraphicsColor", {',
  28107. ' minvalue: -2147483648,',
  28108. ' maxvalue: 2147483647,',
  28109. ' ordtype: 4',
  28110. '});',
  28111. 'this.$rtti.$inherited("TColor", this.$rtti["TGraphicsColor"], {});',
  28112. 'this.p = null;',
  28113. 'this.k = 0;',
  28114. '']),
  28115. LinesToStr([ // $mod.$main
  28116. '$mod.p = $mod.$rtti["TGraphicsColor"];',
  28117. '$mod.p = $mod.$rtti["TColor"];',
  28118. '$mod.k = 1;',
  28119. '$mod.k = 1;',
  28120. '']));
  28121. end;
  28122. procedure TTestModule.TestRTTI_Double;
  28123. begin
  28124. WithTypeInfo:=true;
  28125. StartProgram(true,[supTypeInfo]);
  28126. Add([
  28127. '{$modeswitch externalclass}',
  28128. 'type',
  28129. ' TFloat = type double;',
  28130. 'var',
  28131. ' p: TTypeInfo;',
  28132. 'begin',
  28133. ' p:=typeinfo(double);',
  28134. ' p:=typeinfo(TFloat);',
  28135. '']);
  28136. ConvertProgram;
  28137. CheckSource('TestRTTI_Double',
  28138. LinesToStr([ // statements
  28139. 'this.$rtti.$inherited("TFloat", rtl.double, {});',
  28140. 'this.p = null;',
  28141. '']),
  28142. LinesToStr([ // $mod.$main
  28143. '$mod.p = rtl.double;',
  28144. '$mod.p = $mod.$rtti["TFloat"];',
  28145. '']));
  28146. end;
  28147. procedure TTestModule.TestRTTI_ProcType;
  28148. begin
  28149. WithTypeInfo:=true;
  28150. StartProgram(false);
  28151. Add('type');
  28152. Add(' TProcA = procedure;');
  28153. Add(' TMethodB = procedure of object;');
  28154. Add(' TProcC = procedure; varargs;');
  28155. Add(' TProcD = procedure(i: longint; const j: string; var c: char; out d: double);');
  28156. Add(' TProcE = function: nativeint;');
  28157. Add(' TProcF = function(const p: TProcA): nativeuint;');
  28158. Add('var p: pointer;');
  28159. Add('begin');
  28160. Add(' p:=typeinfo(tproca);');
  28161. ConvertProgram;
  28162. CheckSource('TestRTTI_ProcType',
  28163. LinesToStr([ // statements
  28164. 'this.$rtti.$ProcVar("TProcA", {',
  28165. ' procsig: rtl.newTIProcSig(null)',
  28166. '});',
  28167. 'this.$rtti.$MethodVar("TMethodB", {',
  28168. ' procsig: rtl.newTIProcSig(null),',
  28169. ' methodkind: 0',
  28170. '});',
  28171. 'this.$rtti.$ProcVar("TProcC", {',
  28172. ' procsig: rtl.newTIProcSig(null, 2)',
  28173. '});',
  28174. 'this.$rtti.$ProcVar("TProcD", {',
  28175. ' procsig: rtl.newTIProcSig([["i", rtl.longint], ["j", rtl.string, 2], ["c", rtl.char, 1], ["d", rtl.double, 4]])',
  28176. '});',
  28177. 'this.$rtti.$ProcVar("TProcE", {',
  28178. ' procsig: rtl.newTIProcSig(null, rtl.nativeint)',
  28179. '});',
  28180. 'this.$rtti.$ProcVar("TProcF", {',
  28181. ' procsig: rtl.newTIProcSig([["p", this.$rtti["TProcA"], 2]], rtl.nativeuint)',
  28182. '});',
  28183. 'this.p = null;',
  28184. '']),
  28185. LinesToStr([ // $mod.$main
  28186. '$mod.p = $mod.$rtti["TProcA"];',
  28187. '']));
  28188. end;
  28189. procedure TTestModule.TestRTTI_ProcType_ArgFromOtherUnit;
  28190. begin
  28191. WithTypeInfo:=true;
  28192. AddModuleWithIntfImplSrc('unit2.pas',
  28193. LinesToStr([
  28194. 'type',
  28195. ' TObject = class end;'
  28196. ]),
  28197. '');
  28198. StartUnit(true);
  28199. Add('interface');
  28200. Add('uses unit2;');
  28201. Add('type');
  28202. Add(' TProcA = function(o: tobject): tobject;');
  28203. Add('implementation');
  28204. Add('type');
  28205. Add(' TProcB = function(o: tobject): tobject;');
  28206. Add('var p: Pointer;');
  28207. Add('initialization');
  28208. Add(' p:=typeinfo(tproca);');
  28209. Add(' p:=typeinfo(tprocb);');
  28210. ConvertUnit;
  28211. CheckSource('TestRTTI_ProcType_ArgFromOtherUnit',
  28212. LinesToStr([ // statements
  28213. 'var $impl = $mod.$impl;',
  28214. 'this.$rtti.$ProcVar("TProcA", {',
  28215. ' procsig: rtl.newTIProcSig([["o", pas.unit2.$rtti["TObject"]]], pas.unit2.$rtti["TObject"])',
  28216. '});',
  28217. '']),
  28218. LinesToStr([ // this.$init
  28219. '$impl.p = $mod.$rtti["TProcA"];',
  28220. '$impl.p = $mod.$rtti["TProcB"];',
  28221. '']),
  28222. LinesToStr([ // implementation
  28223. '$mod.$rtti.$ProcVar("TProcB", {',
  28224. ' procsig: rtl.newTIProcSig([["o", pas.unit2.$rtti["TObject"]]], pas.unit2.$rtti["TObject"])',
  28225. '});',
  28226. '$impl.p = null;',
  28227. '']) );
  28228. end;
  28229. procedure TTestModule.TestRTTI_EnumAndSetType;
  28230. begin
  28231. WithTypeInfo:=true;
  28232. StartProgram(false);
  28233. Add('type');
  28234. Add(' TFlag = (light,dark);');
  28235. Add(' TFlags = set of TFlag;');
  28236. Add(' TProc = function(f: TFlags): TFlag;');
  28237. Add('var p: pointer;');
  28238. Add('begin');
  28239. Add(' p:=typeinfo(tflag);');
  28240. Add(' p:=typeinfo(tflags);');
  28241. ConvertProgram;
  28242. CheckSource('TestRTTI_EnumAndType',
  28243. LinesToStr([ // statements
  28244. 'this.TFlag = {',
  28245. ' "0": "light",',
  28246. ' light: 0,',
  28247. ' "1": "dark",',
  28248. ' dark: 1',
  28249. '};',
  28250. 'this.$rtti.$Enum("TFlag", {',
  28251. ' minvalue: 0,',
  28252. ' maxvalue: 1,',
  28253. ' ordtype: 1,',
  28254. ' enumtype: this.TFlag',
  28255. '});',
  28256. 'this.$rtti.$Set("TFlags", {',
  28257. ' comptype: this.$rtti["TFlag"]',
  28258. '});',
  28259. 'this.$rtti.$ProcVar("TProc", {',
  28260. ' procsig: rtl.newTIProcSig([["f", this.$rtti["TFlags"]]], this.$rtti["TFlag"])',
  28261. '});',
  28262. 'this.p = null;',
  28263. '']),
  28264. LinesToStr([ // $mod.$main
  28265. '$mod.p = $mod.$rtti["TFlag"];',
  28266. '$mod.p = $mod.$rtti["TFlags"];',
  28267. '']));
  28268. end;
  28269. procedure TTestModule.TestRTTI_EnumRange;
  28270. begin
  28271. WithTypeInfo:=true;
  28272. StartProgram(false);
  28273. Add([
  28274. 'type',
  28275. ' TCol = (red,green,blue);',
  28276. ' TColRg = green..blue;',
  28277. ' TSetOfColRg = set of TColRg;',
  28278. 'var p: pointer;',
  28279. 'begin',
  28280. ' p:=typeinfo(tcolrg);',
  28281. ' p:=typeinfo(tsetofcolrg);',
  28282. '']);
  28283. ConvertProgram;
  28284. end;
  28285. procedure TTestModule.TestRTTI_AnonymousEnumType;
  28286. begin
  28287. WithTypeInfo:=true;
  28288. StartProgram(false);
  28289. Add('type');
  28290. Add(' TFlags = set of (red, green);');
  28291. Add('var');
  28292. Add(' f: TFlags;');
  28293. Add('begin');
  28294. Add(' Include(f,red);');
  28295. ConvertProgram;
  28296. CheckSource('TestRTTI_AnonymousEnumType',
  28297. LinesToStr([ // statements
  28298. 'this.TFlags$a = {',
  28299. ' "0": "red",',
  28300. ' red: 0,',
  28301. ' "1": "green",',
  28302. ' green: 1',
  28303. '};',
  28304. 'this.$rtti.$Enum("TFlags$a", {',
  28305. ' minvalue: 0,',
  28306. ' maxvalue: 1,',
  28307. ' ordtype: 1,',
  28308. ' enumtype: this.TFlags$a',
  28309. '});',
  28310. 'this.$rtti.$Set("TFlags", {',
  28311. ' comptype: this.$rtti["TFlags$a"]',
  28312. '});',
  28313. 'this.f = {};',
  28314. '']),
  28315. LinesToStr([
  28316. '$mod.f = rtl.includeSet($mod.f, $mod.TFlags$a.red);',
  28317. '']));
  28318. end;
  28319. procedure TTestModule.TestRTTI_StaticArray;
  28320. begin
  28321. WithTypeInfo:=true;
  28322. StartProgram(false);
  28323. Add('type');
  28324. Add(' TFlag = (light,dark);');
  28325. Add(' TFlagNames = array[TFlag] of string;');
  28326. Add(' TBoolNames = array[boolean] of string;');
  28327. Add(' TByteArray = array[1..32768] of byte;');
  28328. Add(' TProc = function(f: TBoolNames): TFlagNames;');
  28329. Add('var p: pointer;');
  28330. Add('begin');
  28331. Add(' p:=typeinfo(TFlagNames);');
  28332. Add(' p:=typeinfo(TBoolNames);');
  28333. ConvertProgram;
  28334. CheckSource('TestRTTI_StaticArray',
  28335. LinesToStr([ // statements
  28336. 'this.TFlag = {',
  28337. ' "0": "light",',
  28338. ' light: 0,',
  28339. ' "1": "dark",',
  28340. ' dark: 1',
  28341. '};',
  28342. 'this.$rtti.$Enum("TFlag", {',
  28343. ' minvalue: 0,',
  28344. ' maxvalue: 1,',
  28345. ' ordtype: 1,',
  28346. ' enumtype: this.TFlag',
  28347. '});',
  28348. 'this.$rtti.$StaticArray("TFlagNames", {',
  28349. ' dims: [2],',
  28350. ' eltype: rtl.string',
  28351. '});',
  28352. 'this.$rtti.$StaticArray("TBoolNames", {',
  28353. ' dims: [2],',
  28354. ' eltype: rtl.string',
  28355. '});',
  28356. 'this.$rtti.$StaticArray("TByteArray", {',
  28357. ' dims: [32768],',
  28358. ' eltype: rtl.byte',
  28359. '});',
  28360. 'this.$rtti.$ProcVar("TProc", {',
  28361. ' procsig: rtl.newTIProcSig([["f", this.$rtti["TBoolNames"]]], this.$rtti["TFlagNames"])',
  28362. '});',
  28363. 'this.p = null;',
  28364. '']),
  28365. LinesToStr([ // $mod.$main
  28366. '$mod.p = $mod.$rtti["TFlagNames"];',
  28367. '$mod.p = $mod.$rtti["TBoolNames"];',
  28368. '']));
  28369. end;
  28370. procedure TTestModule.TestRTTI_DynArray;
  28371. begin
  28372. WithTypeInfo:=true;
  28373. StartProgram(false);
  28374. Add('type');
  28375. Add(' TArrStr = array of string;');
  28376. Add(' TArr2Dim = array of tarrstr;');
  28377. Add(' TProc = function(f: TArrStr): TArr2Dim;');
  28378. Add('var p: pointer;');
  28379. Add('begin');
  28380. Add(' p:=typeinfo(tarrstr);');
  28381. Add(' p:=typeinfo(tarr2dim);');
  28382. ConvertProgram;
  28383. CheckSource('TestRTTI_DynArray',
  28384. LinesToStr([ // statements
  28385. 'this.$rtti.$DynArray("TArrStr", {',
  28386. ' eltype: rtl.string',
  28387. '});',
  28388. 'this.$rtti.$DynArray("TArr2Dim", {',
  28389. ' eltype: this.$rtti["TArrStr"]',
  28390. '});',
  28391. 'this.$rtti.$ProcVar("TProc", {',
  28392. ' procsig: rtl.newTIProcSig([["f", this.$rtti["TArrStr"]]], this.$rtti["TArr2Dim"])',
  28393. '});',
  28394. 'this.p = null;',
  28395. '']),
  28396. LinesToStr([ // $mod.$main
  28397. '$mod.p = $mod.$rtti["TArrStr"];',
  28398. '$mod.p = $mod.$rtti["TArr2Dim"];',
  28399. '']));
  28400. end;
  28401. procedure TTestModule.TestRTTI_ArrayNestedAnonymous;
  28402. begin
  28403. WithTypeInfo:=true;
  28404. StartProgram(false);
  28405. Add('type');
  28406. Add(' TArr = array of array of longint;');
  28407. Add('var a: TArr;');
  28408. Add('begin');
  28409. ConvertProgram;
  28410. CheckSource('TestRTTI_ArrayNestedAnonymous',
  28411. LinesToStr([ // statements
  28412. 'this.$rtti.$DynArray("TArr$a", {',
  28413. ' eltype: rtl.longint',
  28414. '});',
  28415. 'this.$rtti.$DynArray("TArr", {',
  28416. ' eltype: this.$rtti["TArr$a"]',
  28417. '});',
  28418. 'this.a = [];',
  28419. '']),
  28420. LinesToStr([ // $mod.$main
  28421. ]));
  28422. end;
  28423. procedure TTestModule.TestRTTI_PublishedMethodOverloadFail;
  28424. begin
  28425. WithTypeInfo:=true;
  28426. StartProgram(false);
  28427. Add('type');
  28428. Add(' TObject = class');
  28429. Add(' published');
  28430. Add(' procedure Proc; virtual; abstract;');
  28431. Add(' procedure Proc(Sender: tobject); virtual; abstract;');
  28432. Add(' end;');
  28433. Add('begin');
  28434. SetExpectedPasResolverError('Duplicate published method "Proc" at test1.pp(6,19)',
  28435. nDuplicatePublishedMethodXAtY);
  28436. ConvertProgram;
  28437. end;
  28438. procedure TTestModule.TestRTTI_PublishedMethodExternalFail;
  28439. begin
  28440. WithTypeInfo:=true;
  28441. StartProgram(false);
  28442. Add('type');
  28443. Add(' TObject = class');
  28444. Add(' published');
  28445. Add(' procedure Proc; external name ''foo'';');
  28446. Add(' end;');
  28447. Add('begin');
  28448. SetExpectedPasResolverError(sPublishedNameMustMatchExternal,
  28449. nPublishedNameMustMatchExternal);
  28450. ConvertProgram;
  28451. end;
  28452. procedure TTestModule.TestRTTI_PublishedClassPropertyFail;
  28453. begin
  28454. WithTypeInfo:=true;
  28455. StartProgram(false);
  28456. Add('type');
  28457. Add(' TObject = class');
  28458. Add(' class var FA: longint;');
  28459. Add(' published');
  28460. Add(' class property A: longint read FA;');
  28461. Add(' end;');
  28462. Add('begin');
  28463. SetExpectedPasResolverError('Invalid published property modifier "class"',
  28464. nInvalidXModifierY);
  28465. ConvertProgram;
  28466. end;
  28467. procedure TTestModule.TestRTTI_PublishedClassFieldFail;
  28468. begin
  28469. WithTypeInfo:=true;
  28470. StartProgram(false);
  28471. Add('type');
  28472. Add(' TObject = class');
  28473. Add(' published');
  28474. Add(' class var FA: longint;');
  28475. Add(' end;');
  28476. Add('begin');
  28477. SetExpectedPasResolverError(sSymbolCannotBePublished,
  28478. nSymbolCannotBePublished);
  28479. ConvertProgram;
  28480. end;
  28481. procedure TTestModule.TestRTTI_PublishedFieldExternalFail;
  28482. begin
  28483. WithTypeInfo:=true;
  28484. StartProgram(false);
  28485. Add('{$modeswitch externalclass}');
  28486. Add('type');
  28487. Add(' TObject = class');
  28488. Add(' published');
  28489. Add(' V: longint; external name ''foo'';');
  28490. Add(' end;');
  28491. Add('begin');
  28492. SetExpectedPasResolverError(sPublishedNameMustMatchExternal,
  28493. nPublishedNameMustMatchExternal);
  28494. ConvertProgram;
  28495. end;
  28496. procedure TTestModule.TestRTTI_Class_Field;
  28497. begin
  28498. WithTypeInfo:=true;
  28499. StartProgram(false);
  28500. Add('{$modeswitch externalclass}');
  28501. Add('type');
  28502. Add(' TObject = class');
  28503. Add(' private');
  28504. Add(' FPropA: string;');
  28505. Add(' published');
  28506. Add(' VarLI: longint;');
  28507. Add(' VarC: char;');
  28508. Add(' VarS: string;');
  28509. Add(' VarD: double;');
  28510. Add(' VarB: boolean;');
  28511. Add(' VarLW: longword;');
  28512. Add(' VarSmI: smallint;');
  28513. Add(' VarW: word;');
  28514. Add(' VarShI: shortint;');
  28515. Add(' VarBy: byte;');
  28516. Add(' VarExt: longint external name ''VarExt'';');
  28517. Add(' ArrA, ArrB: array of byte;');
  28518. Add(' end;');
  28519. Add('var p: pointer;');
  28520. Add(' Obj: tobject;');
  28521. Add('begin');
  28522. Add(' p:=typeinfo(tobject);');
  28523. Add(' p:=typeinfo(p);');
  28524. Add(' p:=typeinfo(obj);');
  28525. ConvertProgram;
  28526. CheckSource('TestRTTI_Class_Field',
  28527. LinesToStr([ // statements
  28528. 'rtl.createClass(this, "TObject", null, function () {',
  28529. ' this.$init = function () {',
  28530. ' this.FPropA = "";',
  28531. ' this.VarLI = 0;',
  28532. ' this.VarC = "";',
  28533. ' this.VarS = "";',
  28534. ' this.VarD = 0.0;',
  28535. ' this.VarB = false;',
  28536. ' this.VarLW = 0;',
  28537. ' this.VarSmI = 0;',
  28538. ' this.VarW = 0;',
  28539. ' this.VarShI = 0;',
  28540. ' this.VarBy = 0;',
  28541. ' this.ArrA = [];',
  28542. ' this.ArrB = [];',
  28543. ' };',
  28544. ' this.$final = function () {',
  28545. ' this.ArrA = undefined;',
  28546. ' this.ArrB = undefined;',
  28547. ' };',
  28548. ' var $r = this.$rtti;',
  28549. ' $r.addField("VarLI", rtl.longint);',
  28550. ' $r.addField("VarC", rtl.char);',
  28551. ' $r.addField("VarS", rtl.string);',
  28552. ' $r.addField("VarD", rtl.double);',
  28553. ' $r.addField("VarB", rtl.boolean);',
  28554. ' $r.addField("VarLW", rtl.longword);',
  28555. ' $r.addField("VarSmI", rtl.smallint);',
  28556. ' $r.addField("VarW", rtl.word);',
  28557. ' $r.addField("VarShI", rtl.shortint);',
  28558. ' $r.addField("VarBy", rtl.byte);',
  28559. ' $r.addField("VarExt", rtl.longint);',
  28560. ' $mod.$rtti.$DynArray("TObject.ArrB$a", {',
  28561. ' eltype: rtl.byte',
  28562. ' });',
  28563. ' $r.addField("ArrA", $mod.$rtti["TObject.ArrB$a"]);',
  28564. ' $r.addField("ArrB", $mod.$rtti["TObject.ArrB$a"]);',
  28565. '});',
  28566. 'this.p = null;',
  28567. 'this.Obj = null;',
  28568. '']),
  28569. LinesToStr([ // $mod.$main
  28570. '$mod.p = $mod.$rtti["TObject"];',
  28571. '$mod.p = rtl.pointer;',
  28572. '$mod.p = $mod.Obj.$rtti;',
  28573. '']));
  28574. end;
  28575. procedure TTestModule.TestRTTI_Class_Method;
  28576. begin
  28577. WithTypeInfo:=true;
  28578. StartProgram(false);
  28579. Add('type');
  28580. Add(' TObject = class');
  28581. Add(' private');
  28582. Add(' procedure Internal; external name ''$intern'';');
  28583. Add(' published');
  28584. Add(' procedure Click; virtual; abstract;');
  28585. Add(' procedure Notify(Sender: TObject); virtual; abstract;');
  28586. Add(' function GetNotify: boolean; external name ''GetNotify'';');
  28587. Add(' procedure Println(a,b: longint); varargs; virtual; abstract;');
  28588. Add(' end;');
  28589. Add('begin');
  28590. ConvertProgram;
  28591. CheckSource('TestRTTI_Class_Method',
  28592. LinesToStr([ // statements
  28593. 'rtl.createClass(this, "TObject", null, function () {',
  28594. ' this.$init = function () {',
  28595. ' };',
  28596. ' this.$final = function () {',
  28597. ' };',
  28598. ' var $r = this.$rtti;',
  28599. ' $r.addMethod("Click", 0, null);',
  28600. ' $r.addMethod("Notify", 0, [["Sender", $r]]);',
  28601. ' $r.addMethod("GetNotify", 1, null, rtl.boolean,{flags: 4});',
  28602. ' $r.addMethod("Println", 0, [["a", rtl.longint], ["b", rtl.longint]], null, {',
  28603. ' flags: 2',
  28604. ' });',
  28605. '});',
  28606. '']),
  28607. LinesToStr([ // $mod.$main
  28608. '']));
  28609. end;
  28610. procedure TTestModule.TestRTTI_Class_MethodArgFlags;
  28611. begin
  28612. WithTypeInfo:=true;
  28613. StartProgram(false);
  28614. Add('type');
  28615. Add(' TObject = class');
  28616. Add(' published');
  28617. Add(' procedure OpenArray(const Args: array of string); virtual; abstract;');
  28618. Add(' procedure ByRef(var Value: longint; out Item: longint); virtual; abstract;');
  28619. Add(' procedure Untyped(var Value; out Item); virtual; abstract;');
  28620. Add(' end;');
  28621. Add('begin');
  28622. ConvertProgram;
  28623. CheckSource('TestRTTI_Class_MethodOpenArray',
  28624. LinesToStr([ // statements
  28625. 'rtl.createClass(this, "TObject", null, function () {',
  28626. ' this.$init = function () {',
  28627. ' };',
  28628. ' this.$final = function () {',
  28629. ' };',
  28630. ' var $r = this.$rtti;',
  28631. '$r.addMethod("OpenArray", 0, [["Args", rtl.string, 10]]);',
  28632. '$r.addMethod("ByRef", 0, [["Value", rtl.longint, 1], ["Item", rtl.longint, 4]]);',
  28633. '$r.addMethod("Untyped", 0, [["Value", null, 1], ["Item", null, 4]]);',
  28634. '});',
  28635. '']),
  28636. LinesToStr([ // $mod.$main
  28637. '']));
  28638. end;
  28639. procedure TTestModule.TestRTTI_Class_Property;
  28640. begin
  28641. WithTypeInfo:=true;
  28642. StartProgram(false);
  28643. Add('{$modeswitch externalclass}');
  28644. Add('type');
  28645. Add(' TObject = class');
  28646. Add(' private');
  28647. Add(' FColor: longint;');
  28648. Add(' FColorStored: boolean;');
  28649. Add(' procedure SetColor(Value: longint); virtual; abstract;');
  28650. Add(' function GetColor: longint; virtual; abstract;');
  28651. Add(' function GetColorStored: boolean; virtual; abstract;');
  28652. Add(' FExtSize: longint external name ''$extSize'';');
  28653. Add(' FExtSizeStored: boolean external name ''$extSizeStored'';');
  28654. Add(' procedure SetExtSize(Value: longint); external name ''$setSize'';');
  28655. Add(' function GetExtSize: longint; external name ''$getSize'';');
  28656. Add(' function GetExtSizeStored: boolean; external name ''$getExtSizeStored'';');
  28657. Add(' published');
  28658. Add(' property ColorA: longint read FColor;');
  28659. Add(' property ColorB: longint write FColor;');
  28660. Add(' property ColorC: longint read GetColor write SetColor;');
  28661. Add(' property ColorD: longint read FColor write FColor stored FColorStored;');
  28662. Add(' property ExtSizeA: longint read FExtSize write FExtSize;');
  28663. Add(' property ExtSizeB: longint read GetExtSize write SetExtSize stored FExtSizeStored;');
  28664. Add(' property ExtSizeC: longint read FExtSize write FExtSize stored GetExtSizeStored;');
  28665. Add(' end;');
  28666. Add('begin');
  28667. ConvertProgram;
  28668. CheckSource('TestRTTI_Class_Property',
  28669. LinesToStr([ // statements
  28670. 'rtl.createClass(this, "TObject", null, function () {',
  28671. ' this.$init = function () {',
  28672. ' this.FColor = 0;',
  28673. ' this.FColorStored = false;',
  28674. ' };',
  28675. ' this.$final = function () {',
  28676. ' };',
  28677. ' var $r = this.$rtti;',
  28678. ' $r.addProperty("ColorA", 0, rtl.longint, "FColor", "");',
  28679. ' $r.addProperty("ColorB", 0, rtl.longint, "", "FColor");',
  28680. ' $r.addProperty("ColorC", 3, rtl.longint, "GetColor", "SetColor");',
  28681. ' $r.addProperty(',
  28682. ' "ColorD",',
  28683. ' 8,',
  28684. ' rtl.longint,',
  28685. ' "FColor",',
  28686. ' "FColor",',
  28687. ' {',
  28688. ' stored: "FColorStored"',
  28689. ' }',
  28690. ' );',
  28691. ' $r.addProperty("ExtSizeA", 0, rtl.longint, "$extSize", "$extSize");',
  28692. ' $r.addProperty(',
  28693. ' "ExtSizeB",',
  28694. ' 11,',
  28695. ' rtl.longint,',
  28696. ' "$getSize",',
  28697. ' "$setSize",',
  28698. ' {',
  28699. ' stored: "$extSizeStored"',
  28700. ' }',
  28701. ' );',
  28702. ' $r.addProperty(',
  28703. ' "ExtSizeC",',
  28704. ' 12,',
  28705. ' rtl.longint,',
  28706. ' "$extSize",',
  28707. ' "$extSize",',
  28708. ' {',
  28709. ' stored: "$getExtSizeStored"',
  28710. ' }',
  28711. ' );',
  28712. '});',
  28713. '']),
  28714. LinesToStr([ // $mod.$main
  28715. '']));
  28716. end;
  28717. procedure TTestModule.TestRTTI_Class_PropertyParams;
  28718. begin
  28719. WithTypeInfo:=true;
  28720. StartProgram(false);
  28721. Add('{$modeswitch externalclass}');
  28722. Add('type');
  28723. Add(' integer = longint;');
  28724. Add(' TObject = class');
  28725. Add(' private');
  28726. Add(' function GetItems(i: integer): tobject; virtual; abstract;');
  28727. Add(' procedure SetItems(i: integer; value: tobject); virtual; abstract;');
  28728. Add(' function GetValues(const i: integer; var b: boolean): char; virtual; abstract;');
  28729. Add(' procedure SetValues(const i: integer; var b: boolean; value: char); virtual; abstract;');
  28730. Add(' published');
  28731. Add(' property Items[Index: integer]: tobject read getitems write setitems;');
  28732. Add(' property Values[const keya: integer; var keyb: boolean]: char read getvalues write setvalues;');
  28733. Add(' end;');
  28734. Add('begin');
  28735. ConvertProgram;
  28736. CheckSource('TestRTTI_Class_PropertyParams',
  28737. LinesToStr([ // statements
  28738. 'rtl.createClass(this, "TObject", null, function () {',
  28739. ' this.$init = function () {',
  28740. ' };',
  28741. ' this.$final = function () {',
  28742. ' };',
  28743. ' var $r = this.$rtti;',
  28744. ' $r.addProperty("Items", 3, $r, "GetItems", "SetItems");',
  28745. ' $r.addProperty("Values", 3, rtl.char, "GetValues", "SetValues");',
  28746. '});',
  28747. '']),
  28748. LinesToStr([ // $mod.$main
  28749. '']));
  28750. end;
  28751. procedure TTestModule.TestRTTI_Class_OtherUnit_TypeAlias;
  28752. begin
  28753. WithTypeInfo:=true;
  28754. AddModuleWithIntfImplSrc('unit1.pas',
  28755. 'type TColor = -5..5;',
  28756. '');
  28757. StartProgram(true);
  28758. Add([
  28759. 'uses unit1;',
  28760. 'type',
  28761. ' TColorAlias = TColor;',
  28762. ' TColorTypeAlias = type TColor;',
  28763. ' TObject = class',
  28764. ' private',
  28765. ' fColor: TColor;',
  28766. ' fAlias: TColorAlias;',
  28767. ' fTypeAlias: TColorTypeAlias;',
  28768. ' published',
  28769. ' property Color: TColor read fcolor;',
  28770. ' property Alias: TColorAlias read falias;',
  28771. ' property TypeAlias: TColorTypeAlias read ftypealias;',
  28772. ' end;',
  28773. 'begin',
  28774. '']);
  28775. ConvertProgram;
  28776. CheckSource('TestRTTI_Class_OtherUnit_TypeAlias',
  28777. LinesToStr([ // statements
  28778. 'this.$rtti.$inherited("TColorTypeAlias", pas.unit1.$rtti["TColor"], {});',
  28779. 'rtl.createClass(this, "TObject", null, function () {',
  28780. ' this.$init = function () {',
  28781. ' this.fColor = 0;',
  28782. ' this.fAlias = 0;',
  28783. ' this.fTypeAlias = 0;',
  28784. ' };',
  28785. ' this.$final = function () {',
  28786. ' };',
  28787. ' var $r = this.$rtti;',
  28788. ' $r.addProperty("Color", 0, pas.unit1.$rtti["TColor"], "fColor", "");',
  28789. ' $r.addProperty("Alias", 0, pas.unit1.$rtti["TColor"], "fAlias", "");',
  28790. ' $r.addProperty("TypeAlias", 0, $mod.$rtti["TColorTypeAlias"], "fTypeAlias", "");',
  28791. '});',
  28792. '']),
  28793. LinesToStr([ // $mod.$main
  28794. '']));
  28795. end;
  28796. procedure TTestModule.TestRTTI_Class_OmitRTTI;
  28797. begin
  28798. WithTypeInfo:=true;
  28799. StartProgram(false);
  28800. Add([
  28801. '{$modeswitch omitrtti}',
  28802. 'type',
  28803. ' TObject = class',
  28804. ' private',
  28805. ' FA: byte;',
  28806. ' published',
  28807. ' property A: byte read FA write FA;',
  28808. ' end;',
  28809. 'begin']);
  28810. ConvertProgram;
  28811. CheckSource('TestRTTI_Class_OmitRTTI',
  28812. LinesToStr([ // statements
  28813. 'rtl.createClass(this, "TObject", null, function () {',
  28814. ' this.$init = function () {',
  28815. ' this.FA = 0;',
  28816. ' };',
  28817. ' this.$final = function () {',
  28818. ' };',
  28819. '});',
  28820. '']),
  28821. LinesToStr([ // $mod.$main
  28822. '']));
  28823. end;
  28824. procedure TTestModule.TestRTTI_IndexModifier;
  28825. begin
  28826. WithTypeInfo:=true;
  28827. StartProgram(false);
  28828. Add([
  28829. 'type',
  28830. ' TEnum = (red, blue);',
  28831. ' TObject = class',
  28832. ' FB: boolean;',
  28833. ' procedure SetIntBool(Index: longint; b: boolean); virtual; abstract;',
  28834. ' function GetBoolBool(Index: boolean): boolean; virtual; abstract;',
  28835. ' procedure SetBoolBool(Index: boolean; b: boolean); virtual; abstract;',
  28836. ' function GetEnumBool(Index: TEnum): boolean; virtual; abstract;',
  28837. ' function GetStrIntBool(A: String; I: longint): boolean; virtual; abstract;',
  28838. ' procedure SetStrIntBool(A: String; I: longint; b: boolean); virtual; abstract;',
  28839. ' published',
  28840. ' property B1: boolean index 1 read FB write SetIntBool;',
  28841. ' property B2: boolean index TEnum.blue read GetEnumBool write FB;',
  28842. ' property I1[A: String]: boolean index 2 read GetStrIntBool write SetStrIntBool;',
  28843. ' end;',
  28844. 'begin']);
  28845. ConvertProgram;
  28846. CheckSource('TestRTTI_IndexModifier',
  28847. LinesToStr([ // statements
  28848. 'this.TEnum = {',
  28849. ' "0": "red",',
  28850. ' red: 0,',
  28851. ' "1": "blue",',
  28852. ' blue: 1',
  28853. '};',
  28854. 'this.$rtti.$Enum("TEnum", {',
  28855. ' minvalue: 0,',
  28856. ' maxvalue: 1,',
  28857. ' ordtype: 1,',
  28858. ' enumtype: this.TEnum',
  28859. '});',
  28860. 'rtl.createClass(this, "TObject", null, function () {',
  28861. ' this.$init = function () {',
  28862. ' this.FB = false;',
  28863. ' };',
  28864. ' this.$final = function () {',
  28865. ' };',
  28866. ' var $r = this.$rtti;',
  28867. ' $r.addProperty(',
  28868. ' "B1",',
  28869. ' 18,',
  28870. ' rtl.boolean,',
  28871. ' "FB",',
  28872. ' "SetIntBool",',
  28873. ' {',
  28874. ' index: 1',
  28875. ' }',
  28876. ' );',
  28877. ' $r.addProperty(',
  28878. ' "B2",',
  28879. ' 17,',
  28880. ' rtl.boolean,',
  28881. ' "GetEnumBool",',
  28882. ' "FB",',
  28883. ' {',
  28884. ' index: $mod.TEnum.blue',
  28885. ' }',
  28886. ' );',
  28887. ' $r.addProperty(',
  28888. ' "I1",',
  28889. ' 19,',
  28890. ' rtl.boolean,',
  28891. ' "GetStrIntBool",',
  28892. ' "SetStrIntBool",',
  28893. ' {',
  28894. ' index: 2',
  28895. ' }',
  28896. ' );',
  28897. '});',
  28898. '']),
  28899. LinesToStr([ // $mod.$main
  28900. '']));
  28901. end;
  28902. procedure TTestModule.TestRTTI_StoredModifier;
  28903. begin
  28904. WithTypeInfo:=true;
  28905. StartProgram(false);
  28906. Add([
  28907. 'const',
  28908. ' ConstB = true;',
  28909. 'type',
  28910. ' TObject = class',
  28911. ' private',
  28912. ' FB: boolean;',
  28913. ' function IsBStored: boolean; virtual; abstract;',
  28914. ' published',
  28915. ' property BoolA: boolean read FB stored true;',
  28916. ' property BoolB: boolean read FB stored false;',
  28917. ' property BoolC: boolean read FB stored FB;',
  28918. ' property BoolD: boolean read FB stored ConstB;',
  28919. ' property BoolE: boolean read FB stored IsBStored;',
  28920. ' end;',
  28921. 'begin']);
  28922. ConvertProgram;
  28923. CheckSource('TestRTTI_StoredModifier',
  28924. LinesToStr([ // statements
  28925. 'this.ConstB = true;',
  28926. 'rtl.createClass(this, "TObject", null, function () {',
  28927. ' this.$init = function () {',
  28928. ' this.FB = false;',
  28929. ' };',
  28930. ' this.$final = function () {',
  28931. ' };',
  28932. ' var $r = this.$rtti;',
  28933. ' $r.addProperty("BoolA", 0, rtl.boolean, "FB", "");',
  28934. ' $r.addProperty("BoolB", 4, rtl.boolean, "FB", "");',
  28935. ' $r.addProperty(',
  28936. ' "BoolC",',
  28937. ' 8,',
  28938. ' rtl.boolean,',
  28939. ' "FB",',
  28940. ' "",',
  28941. ' {',
  28942. ' stored: "FB"',
  28943. ' }',
  28944. ' );',
  28945. ' $r.addProperty("BoolD", 0, rtl.boolean, "FB", "");',
  28946. ' $r.addProperty(',
  28947. ' "BoolE",',
  28948. ' 12,',
  28949. ' rtl.boolean,',
  28950. ' "FB",',
  28951. ' "",',
  28952. ' {',
  28953. ' stored: "IsBStored"',
  28954. ' }',
  28955. ' );',
  28956. '});',
  28957. '']),
  28958. LinesToStr([ // $mod.$main
  28959. '']));
  28960. end;
  28961. procedure TTestModule.TestRTTI_DefaultValue;
  28962. begin
  28963. WithTypeInfo:=true;
  28964. StartProgram(false);
  28965. Add([
  28966. 'type',
  28967. ' TEnum = (red, blue);',
  28968. 'const',
  28969. ' CB = true or false;',
  28970. ' CI = 1+2;',
  28971. 'type',
  28972. ' TObject = class',
  28973. ' FB: boolean;',
  28974. ' FI: longint;',
  28975. ' FE: TEnum;',
  28976. ' published',
  28977. ' property B1: boolean read FB default true;',
  28978. ' property B2: boolean read FB default CB;',
  28979. ' property B3: boolean read FB default test1.cb;',
  28980. ' property I1: longint read FI default 2;',
  28981. ' property I2: longint read FI default CI;',
  28982. ' property E1: TEnum read FE default red;',
  28983. ' property E2: TEnum read FE default TEnum.blue;',
  28984. ' end;',
  28985. 'begin']);
  28986. ConvertProgram;
  28987. CheckSource('TestRTTI_DefaultValue',
  28988. LinesToStr([ // statements
  28989. 'this.TEnum = {',
  28990. ' "0": "red",',
  28991. ' red: 0,',
  28992. ' "1": "blue",',
  28993. ' blue: 1',
  28994. '};',
  28995. 'this.$rtti.$Enum("TEnum", {',
  28996. ' minvalue: 0,',
  28997. ' maxvalue: 1,',
  28998. ' ordtype: 1,',
  28999. ' enumtype: this.TEnum',
  29000. '});',
  29001. 'this.CB = true || false;',
  29002. 'this.CI = 1 + 2;',
  29003. 'rtl.createClass(this, "TObject", null, function () {',
  29004. ' this.$init = function () {',
  29005. ' this.FB = false;',
  29006. ' this.FI = 0;',
  29007. ' this.FE = 0;',
  29008. ' };',
  29009. ' this.$final = function () {',
  29010. ' };',
  29011. ' var $r = this.$rtti;',
  29012. ' $r.addProperty(',
  29013. ' "B1",',
  29014. ' 0,',
  29015. ' rtl.boolean,',
  29016. ' "FB",',
  29017. ' "",',
  29018. ' {',
  29019. ' Default: true',
  29020. ' }',
  29021. ' );',
  29022. ' $r.addProperty(',
  29023. ' "B2",',
  29024. ' 0,',
  29025. ' rtl.boolean,',
  29026. ' "FB",',
  29027. ' "",',
  29028. ' {',
  29029. ' Default: true',
  29030. ' }',
  29031. ' );',
  29032. ' $r.addProperty(',
  29033. ' "B3",',
  29034. ' 0,',
  29035. ' rtl.boolean,',
  29036. ' "FB",',
  29037. ' "",',
  29038. ' {',
  29039. ' Default: true',
  29040. ' }',
  29041. ' );',
  29042. ' $r.addProperty(',
  29043. ' "I1",',
  29044. ' 0,',
  29045. ' rtl.longint,',
  29046. ' "FI",',
  29047. ' "",',
  29048. ' {',
  29049. ' Default: 2',
  29050. ' }',
  29051. ' );',
  29052. ' $r.addProperty(',
  29053. ' "I2",',
  29054. ' 0,',
  29055. ' rtl.longint,',
  29056. ' "FI",',
  29057. ' "",',
  29058. ' {',
  29059. ' Default: 3',
  29060. ' }',
  29061. ' );',
  29062. ' $r.addProperty(',
  29063. ' "E1",',
  29064. ' 0,',
  29065. ' $mod.$rtti["TEnum"],',
  29066. ' "FE",',
  29067. ' "",',
  29068. ' {',
  29069. ' Default: $mod.TEnum.red',
  29070. ' }',
  29071. ' );',
  29072. ' $r.addProperty(',
  29073. ' "E2",',
  29074. ' 0,',
  29075. ' $mod.$rtti["TEnum"],',
  29076. ' "FE",',
  29077. ' "",',
  29078. ' {',
  29079. ' Default: $mod.TEnum.blue',
  29080. ' }',
  29081. ' );',
  29082. '});',
  29083. '']),
  29084. LinesToStr([ // $mod.$main
  29085. '']));
  29086. end;
  29087. procedure TTestModule.TestRTTI_DefaultValueSet;
  29088. begin
  29089. WithTypeInfo:=true;
  29090. StartProgram(false);
  29091. Add([
  29092. 'type',
  29093. ' TEnum = (red, blue);',
  29094. ' TSet = set of TEnum;',
  29095. 'const',
  29096. ' CSet = [red,blue];',
  29097. 'type',
  29098. ' TObject = class',
  29099. ' FSet: TSet;',
  29100. ' published',
  29101. ' property Set1: TSet read FSet default [];',
  29102. ' property Set2: TSet read FSet default [red];',
  29103. ' property Set3: TSet read FSet default [red,blue];',
  29104. ' property Set4: TSet read FSet default CSet;',
  29105. ' end;',
  29106. 'begin']);
  29107. ConvertProgram;
  29108. CheckSource('TestRTTI_DefaultValueSet',
  29109. LinesToStr([ // statements
  29110. 'this.TEnum = {',
  29111. ' "0": "red",',
  29112. ' red: 0,',
  29113. ' "1": "blue",',
  29114. ' blue: 1',
  29115. '};',
  29116. 'this.$rtti.$Enum("TEnum", {',
  29117. ' minvalue: 0,',
  29118. ' maxvalue: 1,',
  29119. ' ordtype: 1,',
  29120. ' enumtype: this.TEnum',
  29121. '});',
  29122. 'this.$rtti.$Set("TSet", {',
  29123. ' comptype: this.$rtti["TEnum"]',
  29124. '});',
  29125. 'this.CSet = rtl.createSet(this.TEnum.red, this.TEnum.blue);',
  29126. 'rtl.createClass(this, "TObject", null, function () {',
  29127. ' this.$init = function () {',
  29128. ' this.FSet = {};',
  29129. ' };',
  29130. ' this.$final = function () {',
  29131. ' this.FSet = undefined;',
  29132. ' };',
  29133. ' var $r = this.$rtti;',
  29134. ' $r.addProperty(',
  29135. ' "Set1",',
  29136. ' 0,',
  29137. ' $mod.$rtti["TSet"],',
  29138. ' "FSet",',
  29139. ' "",',
  29140. ' {',
  29141. ' Default: {}',
  29142. ' }',
  29143. ' );',
  29144. ' $r.addProperty(',
  29145. ' "Set2",',
  29146. ' 0,',
  29147. ' $mod.$rtti["TSet"],',
  29148. ' "FSet",',
  29149. ' "",',
  29150. ' {',
  29151. ' Default: rtl.createSet($mod.TEnum.red)',
  29152. ' }',
  29153. ' );',
  29154. ' $r.addProperty(',
  29155. ' "Set3",',
  29156. ' 0,',
  29157. ' $mod.$rtti["TSet"],',
  29158. ' "FSet",',
  29159. ' "",',
  29160. ' {',
  29161. ' Default: rtl.createSet($mod.TEnum.red, $mod.TEnum.blue)',
  29162. ' }',
  29163. ' );',
  29164. ' $r.addProperty(',
  29165. ' "Set4",',
  29166. ' 0,',
  29167. ' $mod.$rtti["TSet"],',
  29168. ' "FSet",',
  29169. ' "",',
  29170. ' {',
  29171. ' Default: $mod.CSet',
  29172. ' }',
  29173. ' );',
  29174. '});',
  29175. '']),
  29176. LinesToStr([ // $mod.$main
  29177. '']));
  29178. end;
  29179. procedure TTestModule.TestRTTI_DefaultValueRangeType;
  29180. begin
  29181. WithTypeInfo:=true;
  29182. StartProgram(false);
  29183. Add([
  29184. 'type',
  29185. ' TRg = -1..1;',
  29186. 'const',
  29187. ' l = low(TRg);',
  29188. ' h = high(TRg);',
  29189. 'type',
  29190. ' TObject = class',
  29191. ' FV: TRg;',
  29192. ' published',
  29193. ' property V1: TRg read FV default -1;',
  29194. ' end;',
  29195. 'begin']);
  29196. ConvertProgram;
  29197. CheckSource('TestRTTI_DefaultValueRangeType',
  29198. LinesToStr([ // statements
  29199. 'this.$rtti.$Int("TRg", {',
  29200. ' minvalue: -1,',
  29201. ' maxvalue: 1,',
  29202. ' ordtype: 0',
  29203. '});',
  29204. 'this.l = -1;',
  29205. 'this.h = 1;',
  29206. 'rtl.createClass(this, "TObject", null, function () {',
  29207. ' this.$init = function () {',
  29208. ' this.FV = 0;',
  29209. ' };',
  29210. ' this.$final = function () {',
  29211. ' };',
  29212. ' var $r = this.$rtti;',
  29213. ' $r.addProperty(',
  29214. ' "V1",',
  29215. ' 0,',
  29216. ' $mod.$rtti["TRg"],',
  29217. ' "FV",',
  29218. ' "",',
  29219. ' {',
  29220. ' Default: -1',
  29221. ' }',
  29222. ' );',
  29223. '});',
  29224. '']),
  29225. LinesToStr([ // $mod.$main
  29226. '']));
  29227. end;
  29228. procedure TTestModule.TestRTTI_DefaultValueInherit;
  29229. begin
  29230. WithTypeInfo:=true;
  29231. StartProgram(false);
  29232. Add([
  29233. 'type',
  29234. ' TObject = class',
  29235. ' FA, FB: byte;',
  29236. ' property A: byte read FA default 1;',
  29237. ' property B: byte read FB default 2;',
  29238. ' end;',
  29239. ' TBird = class',
  29240. ' published',
  29241. ' property A;',
  29242. ' property B nodefault;',
  29243. ' end;',
  29244. 'begin']);
  29245. ConvertProgram;
  29246. CheckSource('TestRTTI_DefaultValueInherit',
  29247. LinesToStr([ // statements
  29248. 'rtl.createClass(this, "TObject", null, function () {',
  29249. ' this.$init = function () {',
  29250. ' this.FA = 0;',
  29251. ' this.FB = 0;',
  29252. ' };',
  29253. ' this.$final = function () {',
  29254. ' };',
  29255. '});',
  29256. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  29257. ' var $r = this.$rtti;',
  29258. ' $r.addProperty(',
  29259. ' "A",',
  29260. ' 0,',
  29261. ' rtl.byte,',
  29262. ' "FA",',
  29263. ' "",',
  29264. ' {',
  29265. ' Default: 1',
  29266. ' }',
  29267. ' );',
  29268. ' $r.addProperty("B", 0, rtl.byte, "FB", "");',
  29269. '});',
  29270. '']),
  29271. LinesToStr([ // $mod.$main
  29272. '']));
  29273. end;
  29274. procedure TTestModule.TestRTTI_OverrideMethod;
  29275. begin
  29276. WithTypeInfo:=true;
  29277. StartProgram(false);
  29278. Add('type');
  29279. Add(' TObject = class');
  29280. Add(' published');
  29281. Add(' procedure DoIt; virtual; abstract;');
  29282. Add(' end;');
  29283. Add(' TSky = class');
  29284. Add(' published');
  29285. Add(' procedure DoIt; override;');
  29286. Add(' end;');
  29287. Add('procedure TSky.DoIt; begin end;');
  29288. Add('begin');
  29289. ConvertProgram;
  29290. CheckSource('TestRTTI_OverrideMethod',
  29291. LinesToStr([ // statements
  29292. 'rtl.createClass(this, "TObject", null, function () {',
  29293. ' this.$init = function () {',
  29294. ' };',
  29295. ' this.$final = function () {',
  29296. ' };',
  29297. ' var $r = this.$rtti;',
  29298. ' $r.addMethod("DoIt", 0, null);',
  29299. '});',
  29300. 'rtl.createClass(this, "TSky", this.TObject, function () {',
  29301. ' this.DoIt = function () {',
  29302. ' };',
  29303. '});',
  29304. '']),
  29305. LinesToStr([ // $mod.$main
  29306. '']));
  29307. end;
  29308. procedure TTestModule.TestRTTI_ReintroduceMethod;
  29309. begin
  29310. WithTypeInfo:=true;
  29311. StartProgram(false);
  29312. Add([
  29313. 'type',
  29314. ' TObject = class',
  29315. ' published',
  29316. ' procedure DoIt;',
  29317. ' end;',
  29318. ' TSky = class',
  29319. ' published',
  29320. ' procedure DoIt; reintroduce;',
  29321. ' end;',
  29322. 'procedure TObject.DoIt; begin end;',
  29323. 'procedure TSky.DoIt;',
  29324. 'begin',
  29325. ' inherited DoIt;',
  29326. 'end;',
  29327. 'begin']);
  29328. ConvertProgram;
  29329. CheckSource('TestRTTI_ReintroduceMethod',
  29330. LinesToStr([ // statements
  29331. 'rtl.createClass(this, "TObject", null, function () {',
  29332. ' this.$init = function () {',
  29333. ' };',
  29334. ' this.$final = function () {',
  29335. ' };',
  29336. ' this.DoIt = function () {',
  29337. ' };',
  29338. ' var $r = this.$rtti;',
  29339. ' $r.addMethod("DoIt", 0, null);',
  29340. '});',
  29341. 'rtl.createClass(this, "TSky", this.TObject, function () {',
  29342. ' this.DoIt = function () {',
  29343. ' $mod.TObject.DoIt.call(this);',
  29344. ' };',
  29345. ' var $r = this.$rtti;',
  29346. ' $r.addMethod("DoIt", 0, null);',
  29347. '});',
  29348. '']),
  29349. LinesToStr([ // $mod.$main
  29350. '']));
  29351. end;
  29352. procedure TTestModule.TestRTTI_OverloadProperty;
  29353. begin
  29354. WithTypeInfo:=true;
  29355. StartProgram(false);
  29356. Add('type');
  29357. Add(' TObject = class');
  29358. Add(' protected');
  29359. Add(' FFlag: longint;');
  29360. Add(' published');
  29361. Add(' property Flag: longint read fflag;');
  29362. Add(' end;');
  29363. Add(' TSky = class');
  29364. Add(' published');
  29365. Add(' property FLAG: longint write fflag;');
  29366. Add(' end;');
  29367. Add('begin');
  29368. ConvertProgram;
  29369. CheckSource('TestRTTI_OverrideMethod',
  29370. LinesToStr([ // statements
  29371. 'rtl.createClass(this, "TObject", null, function () {',
  29372. ' this.$init = function () {',
  29373. ' this.FFlag = 0;',
  29374. ' };',
  29375. ' this.$final = function () {',
  29376. ' };',
  29377. ' var $r = this.$rtti;',
  29378. ' $r.addProperty("Flag", 0, rtl.longint, "FFlag", "");',
  29379. '});',
  29380. 'rtl.createClass(this, "TSky", this.TObject, function () {',
  29381. ' var $r = this.$rtti;',
  29382. ' $r.addProperty("Flag", 0, rtl.longint, "", "FFlag");',
  29383. '});',
  29384. '']),
  29385. LinesToStr([ // $mod.$main
  29386. '']));
  29387. end;
  29388. procedure TTestModule.TestRTTI_ClassForward;
  29389. begin
  29390. WithTypeInfo:=true;
  29391. StartProgram(false);
  29392. Add('type');
  29393. Add(' TObject = class end;');
  29394. Add(' tbridge = class;');
  29395. Add(' TProc = function: tbridge;');
  29396. Add(' TOger = class');
  29397. Add(' published');
  29398. Add(' FBridge: tbridge;');
  29399. Add(' procedure SetBridge(Value: tbridge); virtual; abstract;');
  29400. Add(' property Bridge: tbridge read fbridge write setbridge;');
  29401. Add(' end;');
  29402. Add(' TBridge = class');
  29403. Add(' FOger: toger;');
  29404. Add(' end;');
  29405. Add('var p: Pointer;');
  29406. Add(' b: tbridge;');
  29407. Add('begin');
  29408. Add(' p:=typeinfo(tbridge);');
  29409. Add(' p:=typeinfo(b);');
  29410. ConvertProgram;
  29411. CheckSource('TestRTTI_ClassForward',
  29412. LinesToStr([ // statements
  29413. 'rtl.createClass(this, "TObject", null, function () {',
  29414. ' this.$init = function () {',
  29415. ' };',
  29416. ' this.$final = function () {',
  29417. ' };',
  29418. '});',
  29419. 'this.$rtti.$Class("TBridge");',
  29420. 'this.$rtti.$ProcVar("TProc", {',
  29421. ' procsig: rtl.newTIProcSig(null, this.$rtti["TBridge"])',
  29422. '});',
  29423. 'rtl.createClass(this, "TOger", this.TObject, function () {',
  29424. ' this.$init = function () {',
  29425. ' $mod.TObject.$init.call(this);',
  29426. ' this.FBridge = null;',
  29427. ' };',
  29428. ' this.$final = function () {',
  29429. ' this.FBridge = undefined;',
  29430. ' $mod.TObject.$final.call(this);',
  29431. ' };',
  29432. ' var $r = this.$rtti;',
  29433. ' $r.addField("FBridge", $mod.$rtti["TBridge"]);',
  29434. ' $r.addMethod("SetBridge", 0, [["Value", $mod.$rtti["TBridge"]]]);',
  29435. ' $r.addProperty("Bridge", 2, $mod.$rtti["TBridge"], "FBridge", "SetBridge");',
  29436. '});',
  29437. 'rtl.createClass(this, "TBridge", this.TObject, function () {',
  29438. ' this.$init = function () {',
  29439. ' $mod.TObject.$init.call(this);',
  29440. ' this.FOger = null;',
  29441. ' };',
  29442. ' this.$final = function () {',
  29443. ' this.FOger = undefined;',
  29444. ' $mod.TObject.$final.call(this);',
  29445. ' };',
  29446. '});',
  29447. 'this.p = null;',
  29448. 'this.b = null;',
  29449. '']),
  29450. LinesToStr([ // $mod.$main
  29451. '$mod.p = $mod.$rtti["TBridge"];',
  29452. '$mod.p = $mod.b.$rtti;',
  29453. '']));
  29454. end;
  29455. procedure TTestModule.TestRTTI_ClassOf;
  29456. begin
  29457. WithTypeInfo:=true;
  29458. StartProgram(false);
  29459. Add('type');
  29460. Add(' TClass = class of tobject;');
  29461. Add(' TProcA = function: TClass;');
  29462. Add(' TObject = class');
  29463. Add(' published');
  29464. Add(' C: tclass;');
  29465. Add(' end;');
  29466. Add(' tfox = class;');
  29467. Add(' TBird = class end;');
  29468. Add(' TBirds = class of tbird;');
  29469. Add(' TFox = class end;');
  29470. Add(' TFoxes = class of tfox;');
  29471. Add(' TCows = class of TCow;');
  29472. Add(' TCow = class;');
  29473. Add(' TCow = class end;');
  29474. Add('begin');
  29475. ConvertProgram;
  29476. CheckSource('TestRTTI_ClassOf',
  29477. LinesToStr([ // statements
  29478. 'this.$rtti.$Class("TObject");',
  29479. 'this.$rtti.$ClassRef("TClass", {',
  29480. ' instancetype: this.$rtti["TObject"]',
  29481. '});',
  29482. 'this.$rtti.$ProcVar("TProcA", {',
  29483. ' procsig: rtl.newTIProcSig(null, this.$rtti["TClass"])',
  29484. '});',
  29485. 'rtl.createClass(this, "TObject", null, function () {',
  29486. ' this.$init = function () {',
  29487. ' this.C = null;',
  29488. ' };',
  29489. ' this.$final = function () {',
  29490. ' this.C = undefined;',
  29491. ' };',
  29492. ' var $r = this.$rtti;',
  29493. ' $r.addField("C", $mod.$rtti["TClass"]);',
  29494. '});',
  29495. 'this.$rtti.$Class("TFox");',
  29496. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  29497. '});',
  29498. 'this.$rtti.$ClassRef("TBirds", {',
  29499. ' instancetype: this.$rtti["TBird"]',
  29500. '});',
  29501. 'rtl.createClass(this, "TFox", this.TObject, function () {',
  29502. '});',
  29503. 'this.$rtti.$ClassRef("TFoxes", {',
  29504. ' instancetype: this.$rtti["TFox"]',
  29505. '});',
  29506. 'this.$rtti.$Class("TCow");',
  29507. 'this.$rtti.$ClassRef("TCows", {',
  29508. ' instancetype: this.$rtti["TCow"]',
  29509. '});',
  29510. 'rtl.createClass(this, "TCow", this.TObject, function () {',
  29511. '});',
  29512. '']),
  29513. LinesToStr([ // $mod.$main
  29514. '']));
  29515. end;
  29516. procedure TTestModule.TestRTTI_Record;
  29517. begin
  29518. WithTypeInfo:=true;
  29519. StartProgram(false);
  29520. Add('type');
  29521. Add(' integer = longint;');
  29522. Add(' TPoint = record');
  29523. Add(' x,y: integer;');
  29524. Add(' end;');
  29525. Add('var p: pointer;');
  29526. Add(' r: tpoint;');
  29527. Add('begin');
  29528. Add(' p:=typeinfo(tpoint);');
  29529. Add(' p:=typeinfo(r);');
  29530. Add(' p:=typeinfo(r.x);');
  29531. ConvertProgram;
  29532. CheckSource('TestRTTI_Record',
  29533. LinesToStr([ // statements
  29534. 'rtl.recNewT(this, "TPoint", function () {',
  29535. ' this.x = 0;',
  29536. ' this.y = 0;',
  29537. ' this.$eq = function (b) {',
  29538. ' return (this.x === b.x) && (this.y === b.y);',
  29539. ' };',
  29540. ' this.$assign = function (s) {',
  29541. ' this.x = s.x;',
  29542. ' this.y = s.y;',
  29543. ' return this;',
  29544. ' };',
  29545. ' var $r = $mod.$rtti.$Record("TPoint", {});',
  29546. ' $r.addField("x", rtl.longint);',
  29547. ' $r.addField("y", rtl.longint);',
  29548. '});',
  29549. 'this.p = null;',
  29550. 'this.r = this.TPoint.$new();',
  29551. '']),
  29552. LinesToStr([ // $mod.$main
  29553. '$mod.p = $mod.$rtti["TPoint"];',
  29554. '$mod.p = $mod.$rtti["TPoint"];',
  29555. '$mod.p = rtl.longint;',
  29556. '']));
  29557. end;
  29558. procedure TTestModule.TestRTTI_RecordAnonymousArray;
  29559. begin
  29560. WithTypeInfo:=true;
  29561. StartProgram(false);
  29562. Add('type');
  29563. Add(' TFloatRec = record');
  29564. Add(' c,d: array of char;');
  29565. // Add(' i: array of array of longint;');
  29566. Add(' end;');
  29567. Add('var p: pointer;');
  29568. Add(' r: tfloatrec;');
  29569. Add('begin');
  29570. Add(' p:=typeinfo(tfloatrec);');
  29571. Add(' p:=typeinfo(r);');
  29572. Add(' p:=typeinfo(r.d);');
  29573. ConvertProgram;
  29574. CheckSource('TestRTTI_Record',
  29575. LinesToStr([ // statements
  29576. 'rtl.recNewT(this, "TFloatRec", function () {',
  29577. ' this.$new = function () {',
  29578. ' var r = Object.create(this);',
  29579. ' r.c = [];',
  29580. ' r.d = [];',
  29581. ' return r;',
  29582. ' };',
  29583. ' this.$eq = function (b) {',
  29584. ' return (this.c === b.c) && (this.d === b.d);',
  29585. ' };',
  29586. ' this.$assign = function (s) {',
  29587. ' this.c = rtl.arrayRef(s.c);',
  29588. ' this.d = rtl.arrayRef(s.d);',
  29589. ' return this;',
  29590. ' };',
  29591. ' $mod.$rtti.$DynArray("TFloatRec.d$a", {',
  29592. ' eltype: rtl.char',
  29593. ' });',
  29594. ' var $r = $mod.$rtti.$Record("TFloatRec", {});',
  29595. ' $r.addField("c", $mod.$rtti["TFloatRec.d$a"]);',
  29596. ' $r.addField("d", $mod.$rtti["TFloatRec.d$a"]);',
  29597. '});',
  29598. 'this.p = null;',
  29599. 'this.r = this.TFloatRec.$new();',
  29600. '']),
  29601. LinesToStr([ // $mod.$main
  29602. '$mod.p = $mod.$rtti["TFloatRec"];',
  29603. '$mod.p = $mod.$rtti["TFloatRec"];',
  29604. '$mod.p = $mod.$rtti["TFloatRec.d$a"];',
  29605. '']));
  29606. end;
  29607. procedure TTestModule.TestRTTI_LocalTypes;
  29608. begin
  29609. WithTypeInfo:=true;
  29610. StartProgram(false);
  29611. Add([
  29612. 'procedure DoIt;',
  29613. 'type',
  29614. ' integer = longint;',
  29615. ' TPoint = record',
  29616. ' x,y: integer;',
  29617. ' end;',
  29618. 'var p: TPoint;',
  29619. 'begin',
  29620. 'end;',
  29621. 'begin']);
  29622. ConvertProgram;
  29623. CheckSource('TestRTTI_LocalTypes',
  29624. LinesToStr([ // statements
  29625. 'var TPoint = rtl.recNewT(null, "", function () {',
  29626. ' this.x = 0;',
  29627. ' this.y = 0;',
  29628. ' this.$eq = function (b) {',
  29629. ' return (this.x === b.x) && (this.y === b.y);',
  29630. ' };',
  29631. ' this.$assign = function (s) {',
  29632. ' this.x = s.x;',
  29633. ' this.y = s.y;',
  29634. ' return this;',
  29635. ' };',
  29636. '});',
  29637. 'this.DoIt = function () {',
  29638. ' var p = TPoint.$new();',
  29639. '};',
  29640. '']),
  29641. LinesToStr([ // $mod.$main
  29642. '']));
  29643. end;
  29644. procedure TTestModule.TestRTTI_TypeInfo_BaseTypes;
  29645. begin
  29646. WithTypeInfo:=true;
  29647. StartProgram(false);
  29648. Add([
  29649. 'type',
  29650. ' TCaption = string;',
  29651. ' TYesNo = boolean;',
  29652. ' TLetter = char;',
  29653. ' TFloat = double;',
  29654. ' TPtr = pointer;',
  29655. ' TShortInt = shortint;',
  29656. ' TByte = byte;',
  29657. ' TSmallInt = smallint;',
  29658. ' TWord = word;',
  29659. ' TInt32 = longint;',
  29660. ' TDWord = longword;',
  29661. ' TValue = jsvalue;',
  29662. 'var p: TPtr;',
  29663. 'begin',
  29664. ' p:=typeinfo(string);',
  29665. ' p:=typeinfo(tcaption);',
  29666. ' p:=typeinfo(boolean);',
  29667. ' p:=typeinfo(tyesno);',
  29668. ' p:=typeinfo(char);',
  29669. ' p:=typeinfo(tletter);',
  29670. ' p:=typeinfo(double);',
  29671. ' p:=typeinfo(tfloat);',
  29672. ' p:=typeinfo(pointer);',
  29673. ' p:=typeinfo(tptr);',
  29674. ' p:=typeinfo(shortint);',
  29675. ' p:=typeinfo(tshortint);',
  29676. ' p:=typeinfo(byte);',
  29677. ' p:=typeinfo(tbyte);',
  29678. ' p:=typeinfo(smallint);',
  29679. ' p:=typeinfo(tsmallint);',
  29680. ' p:=typeinfo(word);',
  29681. ' p:=typeinfo(tword);',
  29682. ' p:=typeinfo(longword);',
  29683. ' p:=typeinfo(tdword);',
  29684. ' p:=typeinfo(jsvalue);',
  29685. ' p:=typeinfo(tvalue);',
  29686. '']);
  29687. ConvertProgram;
  29688. CheckSource('TestRTTI_TypeInfo_BaseTypes',
  29689. LinesToStr([ // statements
  29690. 'this.p = null;',
  29691. '']),
  29692. LinesToStr([ // $mod.$main
  29693. '$mod.p = rtl.string;',
  29694. '$mod.p = rtl.string;',
  29695. '$mod.p = rtl.boolean;',
  29696. '$mod.p = rtl.boolean;',
  29697. '$mod.p = rtl.char;',
  29698. '$mod.p = rtl.char;',
  29699. '$mod.p = rtl.double;',
  29700. '$mod.p = rtl.double;',
  29701. '$mod.p = rtl.pointer;',
  29702. '$mod.p = rtl.pointer;',
  29703. '$mod.p = rtl.shortint;',
  29704. '$mod.p = rtl.shortint;',
  29705. '$mod.p = rtl.byte;',
  29706. '$mod.p = rtl.byte;',
  29707. '$mod.p = rtl.smallint;',
  29708. '$mod.p = rtl.smallint;',
  29709. '$mod.p = rtl.word;',
  29710. '$mod.p = rtl.word;',
  29711. '$mod.p = rtl.longword;',
  29712. '$mod.p = rtl.longword;',
  29713. '$mod.p = rtl.jsvalue;',
  29714. '$mod.p = rtl.jsvalue;',
  29715. '']));
  29716. end;
  29717. procedure TTestModule.TestRTTI_TypeInfo_Type_BaseTypes;
  29718. begin
  29719. WithTypeInfo:=true;
  29720. StartProgram(false);
  29721. Add([
  29722. 'type',
  29723. ' TCaption = type string;',
  29724. ' TYesNo = type boolean;',
  29725. ' TLetter = type char;',
  29726. ' TFloat = type double;',
  29727. ' TPtr = type pointer;',
  29728. ' TShortInt = type shortint;',
  29729. ' TByte = type byte;',
  29730. ' TSmallInt = type smallint;',
  29731. ' TWord = type word;',
  29732. ' TInt32 = type longint;',
  29733. ' TDWord = type longword;',
  29734. ' TValue = type jsvalue;',
  29735. ' TAliasValue = type TValue;',
  29736. 'var',
  29737. ' p: TPtr;',
  29738. ' a: TAliasValue;',
  29739. 'begin',
  29740. ' p:=typeinfo(tcaption);',
  29741. ' p:=typeinfo(tyesno);',
  29742. ' p:=typeinfo(tletter);',
  29743. ' p:=typeinfo(tfloat);',
  29744. ' p:=typeinfo(tptr);',
  29745. ' p:=typeinfo(tshortint);',
  29746. ' p:=typeinfo(tbyte);',
  29747. ' p:=typeinfo(tsmallint);',
  29748. ' p:=typeinfo(tword);',
  29749. ' p:=typeinfo(tdword);',
  29750. ' p:=typeinfo(tvalue);',
  29751. ' p:=typeinfo(taliasvalue);',
  29752. ' p:=typeinfo(a);',
  29753. '']);
  29754. ConvertProgram;
  29755. CheckSource('TestRTTI_TypeInfo_Type_BaseTypes',
  29756. LinesToStr([ // statements
  29757. 'this.$rtti.$inherited("TCaption", rtl.string, {});',
  29758. 'this.$rtti.$inherited("TYesNo", rtl.boolean, {});',
  29759. 'this.$rtti.$inherited("TLetter", rtl.char, {});',
  29760. 'this.$rtti.$inherited("TFloat", rtl.double, {});',
  29761. 'this.$rtti.$inherited("TPtr", rtl.pointer, {});',
  29762. 'this.$rtti.$inherited("TShortInt", rtl.shortint, {});',
  29763. 'this.$rtti.$inherited("TByte", rtl.byte, {});',
  29764. 'this.$rtti.$inherited("TSmallInt", rtl.smallint, {});',
  29765. 'this.$rtti.$inherited("TWord", rtl.word, {});',
  29766. 'this.$rtti.$inherited("TInt32", rtl.longint, {});',
  29767. 'this.$rtti.$inherited("TDWord", rtl.longword, {});',
  29768. 'this.$rtti.$inherited("TValue", rtl.jsvalue, {});',
  29769. 'this.$rtti.$inherited("TAliasValue", this.$rtti["TValue"], {});',
  29770. 'this.p = null;',
  29771. 'this.a = undefined;',
  29772. '']),
  29773. LinesToStr([ // $mod.$main
  29774. '$mod.p = $mod.$rtti["TCaption"];',
  29775. '$mod.p = $mod.$rtti["TYesNo"];',
  29776. '$mod.p = $mod.$rtti["TLetter"];',
  29777. '$mod.p = $mod.$rtti["TFloat"];',
  29778. '$mod.p = $mod.$rtti["TPtr"];',
  29779. '$mod.p = $mod.$rtti["TShortInt"];',
  29780. '$mod.p = $mod.$rtti["TByte"];',
  29781. '$mod.p = $mod.$rtti["TSmallInt"];',
  29782. '$mod.p = $mod.$rtti["TWord"];',
  29783. '$mod.p = $mod.$rtti["TDWord"];',
  29784. '$mod.p = $mod.$rtti["TValue"];',
  29785. '$mod.p = $mod.$rtti["TAliasValue"];',
  29786. '$mod.p = $mod.$rtti["TAliasValue"];',
  29787. '']));
  29788. end;
  29789. procedure TTestModule.TestRTTI_TypeInfo_LocalFail;
  29790. begin
  29791. WithTypeInfo:=true;
  29792. StartProgram(false);
  29793. Add('procedure DoIt;');
  29794. Add('type');
  29795. Add(' integer = longint;');
  29796. Add(' TPoint = record');
  29797. Add(' x,y: integer;');
  29798. Add(' end;');
  29799. Add('var p: pointer;');
  29800. Add('begin');
  29801. Add(' p:=typeinfo(tpoint);');
  29802. Add('end;');
  29803. Add('begin');
  29804. SetExpectedPasResolverError(sSymbolCannotBePublished,nSymbolCannotBePublished);
  29805. ConvertProgram;
  29806. end;
  29807. procedure TTestModule.TestRTTI_TypeInfo_ExtTypeInfoClasses1;
  29808. begin
  29809. WithTypeInfo:=true;
  29810. StartProgram(true,[supTypeInfo]);
  29811. Add([
  29812. '{$modeswitch externalclass}',
  29813. 'type',
  29814. ' TFlag = (up,down);',
  29815. ' TFlags = set of TFlag;',
  29816. 'var',
  29817. ' ti: TTypeInfo;',
  29818. ' tiInt: TTypeInfoInteger;',
  29819. ' tiEnum: TTypeInfoEnum;',
  29820. ' tiSet: TTypeInfoSet;',
  29821. 'begin',
  29822. ' ti:=typeinfo(string);',
  29823. ' ti:=typeinfo(boolean);',
  29824. ' ti:=typeinfo(char);',
  29825. ' ti:=typeinfo(double);',
  29826. ' tiInt:=typeinfo(shortint);',
  29827. ' tiInt:=typeinfo(byte);',
  29828. ' tiInt:=typeinfo(smallint);',
  29829. ' tiInt:=typeinfo(word);',
  29830. ' tiInt:=typeinfo(longint);',
  29831. ' tiInt:=typeinfo(longword);',
  29832. ' ti:=typeinfo(jsvalue);',
  29833. ' tiEnum:=typeinfo(tflag);',
  29834. ' tiSet:=typeinfo(tflags);']);
  29835. ConvertProgram;
  29836. CheckSource('TestRTTI_TypeInfo_ExtTypeInfoClasses1',
  29837. LinesToStr([ // statements
  29838. 'this.TFlag = {',
  29839. ' "0": "up",',
  29840. ' up: 0,',
  29841. ' "1": "down",',
  29842. ' down: 1',
  29843. '};',
  29844. 'this.$rtti.$Enum("TFlag", {',
  29845. ' minvalue: 0,',
  29846. ' maxvalue: 1,',
  29847. ' ordtype: 1,',
  29848. ' enumtype: this.TFlag',
  29849. '});',
  29850. 'this.$rtti.$Set("TFlags", {',
  29851. ' comptype: this.$rtti["TFlag"]',
  29852. '});',
  29853. 'this.ti = null;',
  29854. 'this.tiInt = null;',
  29855. 'this.tiEnum = null;',
  29856. 'this.tiSet = null;',
  29857. '']),
  29858. LinesToStr([ // $mod.$main
  29859. '$mod.ti = rtl.string;',
  29860. '$mod.ti = rtl.boolean;',
  29861. '$mod.ti = rtl.char;',
  29862. '$mod.ti = rtl.double;',
  29863. '$mod.tiInt = rtl.shortint;',
  29864. '$mod.tiInt = rtl.byte;',
  29865. '$mod.tiInt = rtl.smallint;',
  29866. '$mod.tiInt = rtl.word;',
  29867. '$mod.tiInt = rtl.longint;',
  29868. '$mod.tiInt = rtl.longword;',
  29869. '$mod.ti = rtl.jsvalue;',
  29870. '$mod.tiEnum = $mod.$rtti["TFlag"];',
  29871. '$mod.tiSet = $mod.$rtti["TFlags"];',
  29872. '']));
  29873. end;
  29874. procedure TTestModule.TestRTTI_TypeInfo_ExtTypeInfoClasses2;
  29875. begin
  29876. WithTypeInfo:=true;
  29877. StartProgram(true,[supTypeInfo]);
  29878. Add('{$modeswitch externalclass}');
  29879. Add('type');
  29880. Add(' TStaticArr = array[boolean] of string;');
  29881. Add(' TDynArr = array of string;');
  29882. Add(' TProc = procedure;');
  29883. Add(' TMethod = procedure of object;');
  29884. Add('var');
  29885. Add(' StaticArray: TStaticArr;');
  29886. Add(' tiStaticArray: TTypeInfoStaticArray;');
  29887. Add(' DynArray: TDynArr;');
  29888. Add(' tiDynArray: TTypeInfoDynArray;');
  29889. Add(' ProcVar: TProc;');
  29890. Add(' tiProcVar: TTypeInfoProcVar;');
  29891. Add(' MethodVar: TMethod;');
  29892. Add(' tiMethodVar: TTypeInfoMethodVar;');
  29893. Add('begin');
  29894. Add(' tiStaticArray:=typeinfo(StaticArray);');
  29895. Add(' tiStaticArray:=typeinfo(TStaticArr);');
  29896. Add(' tiDynArray:=typeinfo(DynArray);');
  29897. Add(' tiDynArray:=typeinfo(TDynArr);');
  29898. Add(' tiProcVar:=typeinfo(ProcVar);');
  29899. Add(' tiProcVar:=typeinfo(TProc);');
  29900. Add(' tiMethodVar:=typeinfo(MethodVar);');
  29901. Add(' tiMethodVar:=typeinfo(TMethod);');
  29902. ConvertProgram;
  29903. CheckSource('TestRTTI_TypeInfo_ExtTypeInfoClasses2',
  29904. LinesToStr([ // statements
  29905. 'this.$rtti.$StaticArray("TStaticArr", {',
  29906. ' dims: [2],',
  29907. ' eltype: rtl.string',
  29908. '});',
  29909. 'this.$rtti.$DynArray("TDynArr", {',
  29910. ' eltype: rtl.string',
  29911. '});',
  29912. 'this.$rtti.$ProcVar("TProc", {',
  29913. ' procsig: rtl.newTIProcSig(null)',
  29914. '});',
  29915. 'this.$rtti.$MethodVar("TMethod", {',
  29916. ' procsig: rtl.newTIProcSig(null),',
  29917. ' methodkind: 0',
  29918. '});',
  29919. 'this.StaticArray = rtl.arraySetLength(null,"",2);',
  29920. 'this.tiStaticArray = null;',
  29921. 'this.DynArray = [];',
  29922. 'this.tiDynArray = null;',
  29923. 'this.ProcVar = null;',
  29924. 'this.tiProcVar = null;',
  29925. 'this.MethodVar = null;',
  29926. 'this.tiMethodVar = null;',
  29927. '']),
  29928. LinesToStr([ // $mod.$main
  29929. '$mod.tiStaticArray = $mod.$rtti["TStaticArr"];',
  29930. '$mod.tiStaticArray = $mod.$rtti["TStaticArr"];',
  29931. '$mod.tiDynArray = $mod.$rtti["TDynArr"];',
  29932. '$mod.tiDynArray = $mod.$rtti["TDynArr"];',
  29933. '$mod.tiProcVar = $mod.$rtti["TProc"];',
  29934. '$mod.tiProcVar = $mod.$rtti["TProc"];',
  29935. '$mod.tiMethodVar = $mod.$rtti["TMethod"];',
  29936. '$mod.tiMethodVar = $mod.$rtti["TMethod"];',
  29937. '']));
  29938. end;
  29939. procedure TTestModule.TestRTTI_TypeInfo_ExtTypeInfoClasses3;
  29940. begin
  29941. WithTypeInfo:=true;
  29942. StartProgram(true,[supTypeInfo]);
  29943. Add('{$modeswitch externalclass}');
  29944. Add('type');
  29945. Add(' TRec = record end;');
  29946. // ToDo: ^TRec
  29947. Add(' TObject = class end;');
  29948. Add(' TClass = class of tobject;');
  29949. Add('var');
  29950. Add(' Rec: trec;');
  29951. Add(' tiRecord: ttypeinforecord;');
  29952. Add(' Obj: tobject;');
  29953. Add(' tiClass: ttypeinfoclass;');
  29954. Add(' aClass: tclass;');
  29955. Add(' tiClassRef: ttypeinfoclassref;');
  29956. // ToDo: ^TRec
  29957. Add(' tiPointer: ttypeinfopointer;');
  29958. Add('begin');
  29959. Add(' tirecord:=typeinfo(trec);');
  29960. Add(' tirecord:=typeinfo(trec);');
  29961. Add(' ticlass:=typeinfo(obj);');
  29962. Add(' ticlass:=typeinfo(tobject);');
  29963. Add(' ticlass:=typeinfo(aclass);');
  29964. Add(' ticlassref:=typeinfo(tclass);');
  29965. ConvertProgram;
  29966. CheckSource('TestRTTI_TypeInfo_ExtTypeInfoClasses3',
  29967. LinesToStr([ // statements
  29968. 'rtl.recNewT(this, "TRec", function () {',
  29969. ' this.$eq = function (b) {',
  29970. ' return true;',
  29971. ' };',
  29972. ' this.$assign = function (s) {',
  29973. ' return this;',
  29974. ' };',
  29975. ' $mod.$rtti.$Record("TRec", {});',
  29976. '});',
  29977. 'rtl.createClass(this, "TObject", null, function () {',
  29978. ' this.$init = function () {',
  29979. ' };',
  29980. ' this.$final = function () {',
  29981. ' };',
  29982. '});',
  29983. 'this.$rtti.$ClassRef("TClass", {',
  29984. ' instancetype: this.$rtti["TObject"]',
  29985. '});',
  29986. 'this.Rec = this.TRec.$new();',
  29987. 'this.tiRecord = null;',
  29988. 'this.Obj = null;',
  29989. 'this.tiClass = null;',
  29990. 'this.aClass = null;',
  29991. 'this.tiClassRef = null;',
  29992. 'this.tiPointer = null;',
  29993. '']),
  29994. LinesToStr([ // $mod.$main
  29995. '$mod.tiRecord = $mod.$rtti["TRec"];',
  29996. '$mod.tiRecord = $mod.$rtti["TRec"];',
  29997. '$mod.tiClass = $mod.Obj.$rtti;',
  29998. '$mod.tiClass = $mod.$rtti["TObject"];',
  29999. '$mod.tiClass = $mod.aClass.$rtti;',
  30000. '$mod.tiClassRef = $mod.$rtti["TClass"];',
  30001. '']));
  30002. end;
  30003. procedure TTestModule.TestRTTI_TypeInfo_FunctionClassType;
  30004. begin
  30005. WithTypeInfo:=true;
  30006. StartProgram(true,[supTypeInfo]);
  30007. Add([
  30008. '{$modeswitch externalclass}',
  30009. 'type',
  30010. ' TClass = class of tobject;',
  30011. ' TObject = class',
  30012. ' function MyClass: TClass;',
  30013. ' class function ClassType: TClass;',
  30014. ' end;',
  30015. 'function TObject.MyClass: TClass;',
  30016. 'var t: TTypeInfoClass;',
  30017. 'begin',
  30018. ' t:=TypeInfo(Self);',
  30019. ' t:=TypeInfo(Result);',
  30020. ' t:=TypeInfo(TObject);',
  30021. 'end;',
  30022. 'class function TObject.ClassType: TClass;',
  30023. 'var t: TTypeInfoClass;',
  30024. 'begin',
  30025. ' t:=TypeInfo(Self);',
  30026. ' t:=TypeInfo(Result);',
  30027. 'end;',
  30028. 'var',
  30029. ' Obj: TObject;',
  30030. ' t: TTypeInfoClass;',
  30031. 'begin',
  30032. ' t:=TypeInfo(TObject.ClassType);',
  30033. ' t:=TypeInfo(Obj.ClassType);',
  30034. ' t:=TypeInfo(Obj.MyClass);',
  30035. '']);
  30036. ConvertProgram;
  30037. CheckSource('TestRTTI_TypeInfo_FunctionClassType',
  30038. LinesToStr([ // statements
  30039. 'this.$rtti.$Class("TObject");',
  30040. 'this.$rtti.$ClassRef("TClass", {',
  30041. ' instancetype: this.$rtti["TObject"]',
  30042. '});',
  30043. 'rtl.createClass(this, "TObject", null, function () {',
  30044. ' this.$init = function () {',
  30045. ' };',
  30046. ' this.$final = function () {',
  30047. ' };',
  30048. ' this.MyClass = function () {',
  30049. ' var Result = null;',
  30050. ' var t = null;',
  30051. ' t = this.$rtti;',
  30052. ' t = Result.$rtti;',
  30053. ' t = $mod.$rtti["TObject"];',
  30054. ' return Result;',
  30055. ' };',
  30056. ' this.ClassType = function () {',
  30057. ' var Result = null;',
  30058. ' var t = null;',
  30059. ' t = this.$rtti;',
  30060. ' t = Result.$rtti;',
  30061. ' return Result;',
  30062. ' };',
  30063. '});',
  30064. 'this.Obj = null;',
  30065. 'this.t = null;',
  30066. '']),
  30067. LinesToStr([ // $mod.$main
  30068. '$mod.t = $mod.TObject.ClassType().$rtti;',
  30069. '$mod.t = $mod.Obj.$class.ClassType().$rtti;',
  30070. '$mod.t = $mod.Obj.MyClass().$rtti;',
  30071. '']));
  30072. end;
  30073. procedure TTestModule.TestRTTI_TypeInfo_MixedUnits_PointerAndClass;
  30074. begin
  30075. WithTypeInfo:=true;
  30076. AddModuleWithIntfImplSrc('typinfo.pas',
  30077. LinesToStr([
  30078. '{$modeswitch externalclass}',
  30079. 'type',
  30080. ' TTypeInfo = class external name ''rtl.tTypeInfo'' end;',
  30081. ' TTypeInfoInteger = class external name ''rtl.tTypeInfoInteger''(TTypeInfo) end;',
  30082. '']),
  30083. '');
  30084. AddModuleWithIntfImplSrc('unit2.pas',
  30085. LinesToStr([
  30086. 'uses typinfo;',
  30087. 'type PTypeInfo = TTypeInfo;', // delphi compatibility code
  30088. 'procedure DoPtr(p: PTypeInfo);',
  30089. 'procedure DoInfo(t: TTypeInfo);',
  30090. 'procedure DoInt(t: TTypeInfoInteger);',
  30091. '']),
  30092. LinesToStr([
  30093. 'procedure DoPtr(p: PTypeInfo);',
  30094. 'begin end;',
  30095. 'procedure DoInfo(t: TTypeInfo);',
  30096. 'begin end;',
  30097. 'procedure DoInt(t: TTypeInfoInteger);',
  30098. 'begin end;',
  30099. '']));
  30100. StartUnit(true);
  30101. Add([
  30102. 'interface',
  30103. 'uses unit2;', // does not use unit typinfo
  30104. 'implementation',
  30105. 'var',
  30106. ' i: byte;',
  30107. ' p: pointer;',
  30108. ' t: PTypeInfo;',
  30109. 'initialization',
  30110. ' p:=typeinfo(i);',
  30111. ' t:=typeinfo(i);',
  30112. ' if p=t then ;',
  30113. ' if p=typeinfo(i) then ;',
  30114. ' if typeinfo(i)=p then ;',
  30115. ' if t=typeinfo(i) then ;',
  30116. ' if typeinfo(i)=t then ;',
  30117. ' DoPtr(p);',
  30118. ' DoPtr(t);',
  30119. ' DoPtr(typeinfo(i));',
  30120. ' DoInfo(p);',
  30121. ' DoInfo(t);',
  30122. ' DoInfo(typeinfo(i));',
  30123. ' DoInt(typeinfo(i));',
  30124. '']);
  30125. ConvertUnit;
  30126. CheckSource('TestRTTI_TypeInfo_MixedUnits_PointerAndClass',
  30127. LinesToStr([ // statements
  30128. 'var $impl = $mod.$impl;',
  30129. '']),
  30130. LinesToStr([ // this.$init
  30131. '$impl.p = rtl.byte;',
  30132. '$impl.t = rtl.byte;',
  30133. 'if ($impl.p === $impl.t) ;',
  30134. 'if ($impl.p === rtl.byte) ;',
  30135. 'if (rtl.byte === $impl.p) ;',
  30136. 'if ($impl.t === rtl.byte) ;',
  30137. 'if (rtl.byte === $impl.t) ;',
  30138. 'pas.unit2.DoPtr($impl.p);',
  30139. 'pas.unit2.DoPtr($impl.t);',
  30140. 'pas.unit2.DoPtr(rtl.byte);',
  30141. 'pas.unit2.DoInfo($impl.p);',
  30142. 'pas.unit2.DoInfo($impl.t);',
  30143. 'pas.unit2.DoInfo(rtl.byte);',
  30144. 'pas.unit2.DoInt(rtl.byte);',
  30145. '']),
  30146. LinesToStr([ // implementation
  30147. '$impl.i = 0;',
  30148. '$impl.p = null;',
  30149. '$impl.t = null;',
  30150. '']) );
  30151. end;
  30152. procedure TTestModule.TestRTTI_Interface_Corba;
  30153. begin
  30154. WithTypeInfo:=true;
  30155. StartProgram(true,[supTypeInfo]);
  30156. Add([
  30157. '{$interfaces corba}',
  30158. '{$modeswitch externalclass}',
  30159. 'type',
  30160. ' IUnknown = interface',
  30161. ' end;',
  30162. ' IBird = interface',
  30163. ' function GetItem: longint;',
  30164. ' procedure SetItem(Value: longint);',
  30165. ' property Item: longint read GetItem write SetItem;',
  30166. ' end;',
  30167. 'procedure DoIt(t: TTypeInfoInterface); begin end;',
  30168. 'var',
  30169. ' i: IBird;',
  30170. ' t: TTypeInfoInterface;',
  30171. 'begin',
  30172. ' t:=TypeInfo(IBird);',
  30173. ' t:=TypeInfo(i);',
  30174. ' DoIt(t);',
  30175. ' DoIt(TypeInfo(IBird));',
  30176. '']);
  30177. ConvertProgram;
  30178. CheckSource('TestRTTI_Interface_Corba',
  30179. LinesToStr([ // statements
  30180. 'rtl.createInterface(',
  30181. ' this,',
  30182. ' "IUnknown",',
  30183. ' "{B92D5841-758A-322B-B800-000000000000}",',
  30184. ' [],',
  30185. ' null,',
  30186. ' function () {',
  30187. ' }',
  30188. ');',
  30189. 'rtl.createInterface(',
  30190. ' this,',
  30191. ' "IBird",',
  30192. ' "{D32D5841-6264-3AE3-A2C9-B91CE922C9B9}",',
  30193. ' ["GetItem", "SetItem"],',
  30194. ' null,',
  30195. ' function () {',
  30196. ' var $r = this.$rtti;',
  30197. ' $r.addMethod("GetItem", 1, null, rtl.longint);',
  30198. ' $r.addMethod("SetItem", 0, [["Value", rtl.longint]]);',
  30199. ' $r.addProperty("Item", 3, rtl.longint, "GetItem", "SetItem");',
  30200. ' }',
  30201. ');',
  30202. 'this.DoIt = function (t) {',
  30203. '}; ',
  30204. 'this.i = null;',
  30205. 'this.t = null;',
  30206. '']),
  30207. LinesToStr([ // $mod.$main
  30208. '$mod.t = $mod.$rtti["IBird"];',
  30209. '$mod.t = $mod.i.$rtti;',
  30210. '$mod.DoIt($mod.t);',
  30211. '$mod.DoIt($mod.$rtti["IBird"]);',
  30212. '']));
  30213. end;
  30214. procedure TTestModule.TestRTTI_Interface_COM;
  30215. begin
  30216. WithTypeInfo:=true;
  30217. StartProgram(true,[supTypeInfo]);
  30218. Add([
  30219. '{$interfaces com}',
  30220. '{$modeswitch externalclass}',
  30221. 'type',
  30222. ' TGuid = record end;',
  30223. ' integer = longint;',
  30224. ' IUnknown = interface',
  30225. ' function QueryInterface(const iid: TGuid; out obj): Integer;',
  30226. ' function _AddRef: Integer;',
  30227. ' function _Release: Integer;',
  30228. ' end;',
  30229. ' IBird = interface',
  30230. ' function GetItem: longint;',
  30231. ' procedure SetItem(Value: longint);',
  30232. ' property Item: longint read GetItem write SetItem;',
  30233. ' end;',
  30234. 'var',
  30235. ' i: IBird;',
  30236. ' t: TTypeInfoInterface;',
  30237. 'begin',
  30238. ' t:=TypeInfo(IBird);',
  30239. ' t:=TypeInfo(i);',
  30240. '']);
  30241. ConvertProgram;
  30242. CheckSource('TestRTTI_Interface_COM',
  30243. LinesToStr([ // statements
  30244. 'rtl.recNewT(this, "TGuid", function () {',
  30245. ' this.$eq = function (b) {',
  30246. ' return true;',
  30247. ' };',
  30248. ' this.$assign = function (s) {',
  30249. ' return this;',
  30250. ' };',
  30251. ' $mod.$rtti.$Record("TGuid", {});',
  30252. '});',
  30253. 'rtl.createInterface(',
  30254. ' this,',
  30255. ' "IUnknown",',
  30256. ' "{D7ADB00D-1A9B-3EDC-B123-730E661DDFA9}",',
  30257. ' ["QueryInterface", "_AddRef", "_Release"],',
  30258. ' null,',
  30259. ' function () {',
  30260. ' this.$kind = "com";',
  30261. ' var $r = this.$rtti;',
  30262. ' $r.addMethod("QueryInterface", 1, [["iid", $mod.$rtti["TGuid"], 2], ["obj", null, 4]], rtl.longint);',
  30263. ' $r.addMethod("_AddRef", 1, null, rtl.longint);',
  30264. ' $r.addMethod("_Release", 1, null, rtl.longint);',
  30265. ' }',
  30266. ');',
  30267. 'rtl.createInterface(',
  30268. ' this,',
  30269. ' "IBird",',
  30270. ' "{9CC77572-0E45-3594-9A88-9E8D865C9E0A}",',
  30271. ' ["GetItem", "SetItem"],',
  30272. ' this.IUnknown,',
  30273. ' function () {',
  30274. ' var $r = this.$rtti;',
  30275. ' $r.addMethod("GetItem", 1, null, rtl.longint);',
  30276. ' $r.addMethod("SetItem", 0, [["Value", rtl.longint]]);',
  30277. ' $r.addProperty("Item", 3, rtl.longint, "GetItem", "SetItem");',
  30278. ' }',
  30279. ');',
  30280. 'this.i = null;',
  30281. 'this.t = null;',
  30282. '']),
  30283. LinesToStr([ // $mod.$main
  30284. '$mod.t = $mod.$rtti["IBird"];',
  30285. '$mod.t = $mod.i.$rtti;',
  30286. '']));
  30287. end;
  30288. procedure TTestModule.TestRTTI_ClassHelper;
  30289. begin
  30290. WithTypeInfo:=true;
  30291. StartProgram(true,[supTypeInfo]);
  30292. Add([
  30293. '{$interfaces com}',
  30294. '{$modeswitch externalclass}',
  30295. 'type',
  30296. ' TObject = class',
  30297. ' end;',
  30298. ' THelper = class helper for TObject',
  30299. ' published',
  30300. ' function GetItem: longint;',
  30301. ' property Item: longint read GetItem;',
  30302. ' end;',
  30303. 'function THelper.GetItem: longint;',
  30304. 'begin',
  30305. 'end;',
  30306. 'var',
  30307. ' t: TTypeInfoHelper;',
  30308. 'begin',
  30309. ' t:=TypeInfo(THelper);',
  30310. '']);
  30311. ConvertProgram;
  30312. CheckSource('TestRTTI_ClassHelper',
  30313. LinesToStr([ // statements
  30314. 'rtl.createClass(this, "TObject", null, function () {',
  30315. ' this.$init = function () {',
  30316. ' };',
  30317. ' this.$final = function () {',
  30318. ' };',
  30319. '});',
  30320. 'rtl.createHelper(this, "THelper", null, function () {',
  30321. ' this.GetItem = function () {',
  30322. ' var Result = 0;',
  30323. ' return Result;',
  30324. ' };',
  30325. ' var $r = this.$rtti;',
  30326. ' $r.addMethod("GetItem", 1, null, rtl.longint);',
  30327. ' $r.addProperty("Item", 1, rtl.longint, "GetItem", "");',
  30328. '});',
  30329. 'this.t = null;',
  30330. '']),
  30331. LinesToStr([ // $mod.$main
  30332. '$mod.t = $mod.$rtti["THelper"];',
  30333. '']));
  30334. end;
  30335. procedure TTestModule.TestRTTI_ExternalClass;
  30336. begin
  30337. WithTypeInfo:=true;
  30338. StartProgram(true,[supTypeInfo]);
  30339. Add([
  30340. '{$modeswitch externalclass}',
  30341. 'type',
  30342. ' TJSObject = class external name ''Object''',
  30343. ' end;',
  30344. ' TJSArray = class external name ''Array'' (TJSObject)',
  30345. ' end;',
  30346. 'var',
  30347. ' p: Pointer;',
  30348. ' tc: TTypeInfoExtClass;',
  30349. 'begin',
  30350. ' p:=typeinfo(TJSArray);']);
  30351. ConvertProgram;
  30352. CheckSource('TestRTTI_ExternalClass',
  30353. LinesToStr([ // statements
  30354. 'this.$rtti.$ExtClass("TJSObject", {',
  30355. ' jsclass: "Object"',
  30356. '});',
  30357. 'this.$rtti.$ExtClass("TJSArray", {',
  30358. ' ancestor: this.$rtti["TJSObject"],',
  30359. ' jsclass: "Array"',
  30360. '});',
  30361. 'this.p = null;',
  30362. 'this.tc = null;',
  30363. '']),
  30364. LinesToStr([ // $mod.$main
  30365. '$mod.p = $mod.$rtti["TJSArray"];',
  30366. '']));
  30367. end;
  30368. procedure TTestModule.TestRTTI_Unit;
  30369. begin
  30370. WithTypeInfo:=true;
  30371. AddModuleWithIntfImplSrc('unit2.pas',
  30372. LinesToStr([
  30373. '{$mode delphi}',
  30374. 'type',
  30375. ' TWordArray = array of word;',
  30376. ' TArray<T> = array of T;',
  30377. '']),
  30378. '');
  30379. StartUnit(true,[supTypeInfo,supTInterfacedObject]);
  30380. Add([
  30381. '{$mode delphi}',
  30382. 'interface',
  30383. 'uses unit2;',
  30384. 'type',
  30385. ' IBird = interface',
  30386. ' function Swoop: TWordArray;',
  30387. ' function Glide: TArray<word>;',
  30388. ' end;',
  30389. 'procedure Fly;',
  30390. 'implementation',
  30391. 'procedure Fly;',
  30392. 'var',
  30393. ' ta: tTypeInfoDynArray;',
  30394. ' ti: tTypeInfoInterface;',
  30395. 'begin',
  30396. ' ta:=typeinfo(TWordArray);',
  30397. ' ta:=typeinfo(TArray<word>);',
  30398. ' ti:=typeinfo(IBird);',
  30399. 'end;',
  30400. '']);
  30401. ConvertUnit;
  30402. CheckSource('TestRTTI_ExternalClass',
  30403. LinesToStr([ // statements
  30404. 'rtl.createInterface(',
  30405. ' this,',
  30406. ' "IBird",',
  30407. ' "{3B98AAAC-6116-3E17-AA85-F16786D85B09}",',
  30408. ' ["Swoop", "Glide"],',
  30409. ' pas.system.IUnknown,',
  30410. ' function () {',
  30411. ' var $r = this.$rtti;',
  30412. ' $r.addMethod("Swoop", 1, null, pas.unit2.$rtti["TWordArray"]);',
  30413. ' $r.addMethod("Glide", 1, null, pas.unit2.$rtti["TArray<System.Word>"]);',
  30414. ' }',
  30415. ');',
  30416. 'this.Fly = function () {',
  30417. ' var ta = null;',
  30418. ' var ti = null;',
  30419. ' ta = pas.unit2.$rtti["TWordArray"];',
  30420. ' ta = pas.unit2.$rtti["TArray<System.Word>"];',
  30421. ' ti = $mod.$rtti["IBird"];',
  30422. '};',
  30423. '']),
  30424. LinesToStr([ // $mod.$main
  30425. '']));
  30426. end;
  30427. procedure TTestModule.TestResourcestringProgram;
  30428. begin
  30429. AddModuleWithIntfImplSrc('unit2.pas',
  30430. LinesToStr([
  30431. 'resourcestring Title = ''Nice'';',
  30432. '']),
  30433. '');
  30434. StartProgram(true);
  30435. Add([
  30436. 'uses unit2;',
  30437. 'const Bar = ''bar'';',
  30438. 'resourcestring',
  30439. ' Red = ''red'';',
  30440. ' Foobar = ''fOo''+bar;',
  30441. 'var s: string;',
  30442. ' c: char;',
  30443. 'begin',
  30444. ' s:=red;',
  30445. ' s:=test1.red;',
  30446. ' s:=Title;',
  30447. ' c:=red[1];',
  30448. ' c:=test1.red[2];',
  30449. ' if red=foobar then ;',
  30450. ' if red[3]=red[4] then ;']);
  30451. ConvertProgram;
  30452. CheckSource('TestResourcestringProgram',
  30453. LinesToStr([ // statements
  30454. 'this.Bar = "bar";',
  30455. 'this.s = "";',
  30456. 'this.c = "";',
  30457. '$mod.$resourcestrings = {',
  30458. ' Red: {',
  30459. ' org: "red"',
  30460. ' },',
  30461. ' Foobar: {',
  30462. ' org: "fOobar"',
  30463. ' }',
  30464. '};',
  30465. '']),
  30466. LinesToStr([ // $mod.$main
  30467. '$mod.s = rtl.getResStr($mod, "Red");',
  30468. '$mod.s = rtl.getResStr($mod, "Red");',
  30469. '$mod.s = rtl.getResStr(pas.unit2, "Title");',
  30470. '$mod.c = rtl.getResStr($mod, "Red").charAt(0);',
  30471. '$mod.c = rtl.getResStr($mod, "Red").charAt(1);',
  30472. 'if (rtl.getResStr($mod, "Red") === rtl.getResStr($mod, "Foobar")) ;',
  30473. 'if (rtl.getResStr($mod, "Red").charAt(2) === rtl.getResStr($mod, "Red").charAt(3)) ;',
  30474. '']));
  30475. end;
  30476. procedure TTestModule.TestResourcestringUnit;
  30477. begin
  30478. AddModuleWithIntfImplSrc('unit2.pas',
  30479. LinesToStr([
  30480. 'resourcestring Title = ''Nice'';',
  30481. '']),
  30482. '');
  30483. StartUnit(true);
  30484. Add([
  30485. 'interface',
  30486. 'uses unit2;',
  30487. 'const Red = ''rEd'';',
  30488. 'resourcestring',
  30489. ' Blue = ''blue'';',
  30490. ' NotRed = ''not''+Red;',
  30491. 'var s: string;',
  30492. 'implementation',
  30493. 'resourcestring',
  30494. ' ImplGreen = ''green'';',
  30495. 'initialization',
  30496. ' s:=blue+ImplGreen;',
  30497. ' s:=test1.blue+test1.implgreen;',
  30498. ' s:=blue[1]+implgreen[2];',
  30499. ' s:=Title;',
  30500. '']);
  30501. ConvertUnit;
  30502. CheckSource('TestResourcestringUnit',
  30503. LinesToStr([ // statements
  30504. 'this.Red = "rEd";',
  30505. 'this.s = "";',
  30506. '$mod.$resourcestrings = {',
  30507. ' Blue: {',
  30508. ' org: "blue"',
  30509. ' },',
  30510. ' NotRed: {',
  30511. ' org: "notrEd"',
  30512. ' },',
  30513. ' ImplGreen: {',
  30514. ' org: "green"',
  30515. ' }',
  30516. '};',
  30517. '']),
  30518. LinesToStr([ // $mod.$main
  30519. '$mod.s = rtl.getResStr($mod, "Blue") + rtl.getResStr($mod, "ImplGreen");',
  30520. '$mod.s = rtl.getResStr($mod, "Blue") + rtl.getResStr($mod, "ImplGreen");',
  30521. '$mod.s = rtl.getResStr($mod, "Blue").charAt(0) + rtl.getResStr($mod, "ImplGreen").charAt(1);',
  30522. '$mod.s = rtl.getResStr(pas.unit2, "Title");',
  30523. '']));
  30524. end;
  30525. procedure TTestModule.TestResourcestringImplementation;
  30526. begin
  30527. StartUnit(false);
  30528. Add([
  30529. 'interface',
  30530. 'implementation',
  30531. 'resourcestring',
  30532. ' ImplRed = ''red'';']);
  30533. ConvertUnit;
  30534. CheckSource('TestResourcestringImplementation',
  30535. LinesToStr([ // intf statements
  30536. 'var $impl = $mod.$impl;']),
  30537. LinesToStr([ // $mod.$init
  30538. '']),
  30539. LinesToStr([ // impl statements
  30540. '$mod.$resourcestrings = {',
  30541. ' ImplRed: {',
  30542. ' org: "red"',
  30543. ' }',
  30544. '};',
  30545. '']));
  30546. end;
  30547. procedure TTestModule.TestAttributes_Members;
  30548. begin
  30549. WithTypeInfo:=true;
  30550. StartProgram(false);
  30551. Add([
  30552. '{$modeswitch PrefixedAttributes}',
  30553. 'type',
  30554. ' TObject = class',
  30555. ' constructor Create;',
  30556. ' end;',
  30557. ' TCustomAttribute = class',
  30558. ' constructor Create(Id: word);',
  30559. ' end;',
  30560. ' [Missing]',
  30561. ' TBird = class',
  30562. ' published',
  30563. ' [Tcustom]',
  30564. ' FField: word;',
  30565. ' [tcustom(14)]',
  30566. ' property Size: word read FField;',
  30567. ' [Tcustom(15)]',
  30568. ' procedure Fly; virtual; abstract;',
  30569. ' end;',
  30570. ' TRec = record',
  30571. ' [Tcustom,tcustom(14)]',
  30572. ' Size: word;',
  30573. ' end;',
  30574. 'constructor TObject.Create; begin end;',
  30575. 'constructor TCustomAttribute.Create(Id: word); begin end;',
  30576. 'begin',
  30577. '']);
  30578. ConvertProgram;
  30579. CheckSource('TestAttributes_Members',
  30580. LinesToStr([ // statements
  30581. 'rtl.createClass(this, "TObject", null, function () {',
  30582. ' this.$init = function () {',
  30583. ' };',
  30584. ' this.$final = function () {',
  30585. ' };',
  30586. ' this.Create = function () {',
  30587. ' return this;',
  30588. ' };',
  30589. '});',
  30590. 'rtl.createClass(this, "TCustomAttribute", this.TObject, function () {',
  30591. ' this.Create$1 = function (Id) {',
  30592. ' return this;',
  30593. ' };',
  30594. '});',
  30595. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  30596. ' this.$init = function () {',
  30597. ' $mod.TObject.$init.call(this);',
  30598. ' this.FField = 0;',
  30599. ' };',
  30600. ' var $r = this.$rtti;',
  30601. ' $r.addField("FField", rtl.word, {',
  30602. ' attr: [$mod.TCustomAttribute, "Create"]',
  30603. ' });',
  30604. ' $r.addProperty(',
  30605. ' "Size",',
  30606. ' 0,',
  30607. ' rtl.word,',
  30608. ' "FField",',
  30609. ' "",',
  30610. ' {',
  30611. ' attr: [$mod.TCustomAttribute, "Create$1", [14]]',
  30612. ' }',
  30613. ' );',
  30614. ' $r.addMethod("Fly", 0, null, null, {',
  30615. ' attr: [$mod.TCustomAttribute, "Create$1", [15]]',
  30616. ' });',
  30617. '});',
  30618. 'rtl.recNewT(this, "TRec", function () {',
  30619. ' this.Size = 0;',
  30620. ' this.$eq = function (b) {',
  30621. ' return this.Size === b.Size;',
  30622. ' };',
  30623. ' this.$assign = function (s) {',
  30624. ' this.Size = s.Size;',
  30625. ' return this;',
  30626. ' };',
  30627. ' var $r = $mod.$rtti.$Record("TRec", {});',
  30628. ' $r.addField("Size", rtl.word, {',
  30629. ' attr: [',
  30630. ' $mod.TCustomAttribute,',
  30631. ' "Create",',
  30632. ' $mod.TCustomAttribute,',
  30633. ' "Create$1",',
  30634. ' [14]',
  30635. ' ]',
  30636. ' });',
  30637. '});',
  30638. '']),
  30639. LinesToStr([ // $mod.$main
  30640. '']));
  30641. end;
  30642. procedure TTestModule.TestAttributes_Types;
  30643. begin
  30644. WithTypeInfo:=true;
  30645. StartProgram(false);
  30646. Add([
  30647. '{$modeswitch PrefixedAttributes}',
  30648. 'type',
  30649. ' TObject = class',
  30650. ' constructor Create(Id: word);',
  30651. ' end;',
  30652. ' TCustomAttribute = class',
  30653. ' end;',
  30654. ' [TCustom(1)]',
  30655. ' TMyClass = class',
  30656. ' end;',
  30657. ' [TCustom(2)]',
  30658. ' TRec = record',
  30659. ' end;',
  30660. ' [TCustom(3)]',
  30661. ' TInt = type word;',
  30662. 'constructor TObject.Create(Id: word);',
  30663. 'begin',
  30664. 'end;',
  30665. 'var p: pointer;',
  30666. 'begin',
  30667. ' p:=typeinfo(TMyClass);',
  30668. ' p:=typeinfo(TRec);',
  30669. ' p:=typeinfo(TInt);',
  30670. '']);
  30671. ConvertProgram;
  30672. CheckSource('TestAttributes_Types',
  30673. LinesToStr([ // statements
  30674. 'rtl.createClass(this, "TObject", null, function () {',
  30675. ' this.$init = function () {',
  30676. ' };',
  30677. ' this.$final = function () {',
  30678. ' };',
  30679. ' this.Create = function (Id) {',
  30680. ' return this;',
  30681. ' };',
  30682. '});',
  30683. 'rtl.createClass(this, "TCustomAttribute", this.TObject, function () {',
  30684. '});',
  30685. 'rtl.createClass(this, "TMyClass", this.TObject, function () {',
  30686. ' var $r = this.$rtti;',
  30687. ' $r.attr = [$mod.TCustomAttribute, "Create", [1]];',
  30688. '});',
  30689. 'rtl.recNewT(this, "TRec", function () {',
  30690. ' this.$eq = function (b) {',
  30691. ' return true;',
  30692. ' };',
  30693. ' this.$assign = function (s) {',
  30694. ' return this;',
  30695. ' };',
  30696. ' $mod.$rtti.$Record("TRec", {',
  30697. ' attr: [$mod.TCustomAttribute, "Create", [2]]',
  30698. ' });',
  30699. '});',
  30700. 'this.$rtti.$inherited("TInt", rtl.word, {',
  30701. ' attr: [this.TCustomAttribute, "Create", [3]]',
  30702. '});',
  30703. 'this.p = null;',
  30704. '']),
  30705. LinesToStr([ // $mod.$main
  30706. '$mod.p = $mod.$rtti["TMyClass"];',
  30707. '$mod.p = $mod.$rtti["TRec"];',
  30708. '$mod.p = $mod.$rtti["TInt"];',
  30709. '']));
  30710. end;
  30711. procedure TTestModule.TestAttributes_HelperConstructor_Fail;
  30712. begin
  30713. WithTypeInfo:=true;
  30714. StartProgram(false);
  30715. Add([
  30716. '{$modeswitch PrefixedAttributes}',
  30717. 'type',
  30718. ' TObject = class',
  30719. ' constructor Create;',
  30720. ' end;',
  30721. ' TCustomAttribute = class',
  30722. ' end;',
  30723. ' THelper = class helper for TCustomAttribute',
  30724. ' constructor Create(Id: word);',
  30725. ' end;',
  30726. ' [TCustom(3)]',
  30727. ' TMyInt = word;',
  30728. 'constructor TObject.Create; begin end;',
  30729. 'constructor THelper.Create(Id: word); begin end;',
  30730. 'begin',
  30731. ' if typeinfo(TMyInt)=nil then ;']);
  30732. ConvertProgram;
  30733. end;
  30734. procedure TTestModule.TestAssert;
  30735. begin
  30736. StartProgram(false);
  30737. Add([
  30738. 'procedure DoIt;',
  30739. 'var',
  30740. ' b: boolean;',
  30741. ' s: string;',
  30742. 'begin',
  30743. ' {$Assertions on}',
  30744. ' Assert(b);',
  30745. 'end;',
  30746. 'begin',
  30747. ' DoIt;',
  30748. '']);
  30749. ConvertProgram;
  30750. CheckSource('TestAssert',
  30751. LinesToStr([ // statements
  30752. 'this.DoIt = function () {',
  30753. ' var b = false;',
  30754. ' var s = "";',
  30755. ' if (!b) throw "assert failed";',
  30756. '};',
  30757. '']),
  30758. LinesToStr([ // $mod.$main
  30759. '$mod.DoIt();',
  30760. '']));
  30761. end;
  30762. procedure TTestModule.TestAssert_SysUtils;
  30763. begin
  30764. AddModuleWithIntfImplSrc('SysUtils.pas',
  30765. LinesToStr([
  30766. 'type',
  30767. ' TObject = class',
  30768. ' constructor Create;',
  30769. ' end;',
  30770. ' EAssertionFailed = class',
  30771. ' constructor Create(s: string);',
  30772. ' end;',
  30773. '']),
  30774. LinesToStr([
  30775. 'constructor TObject.Create;',
  30776. 'begin end;',
  30777. 'constructor EAssertionFailed.Create(s: string);',
  30778. 'begin end;',
  30779. '']) );
  30780. StartProgram(true);
  30781. Add([
  30782. 'uses sysutils;',
  30783. 'procedure DoIt;',
  30784. 'var',
  30785. ' b: boolean;',
  30786. ' s: string;',
  30787. 'begin',
  30788. ' {$Assertions on}',
  30789. ' Assert(b);',
  30790. ' Assert(b,''msg'');',
  30791. 'end;',
  30792. 'begin',
  30793. ' DoIt;',
  30794. '']);
  30795. ConvertProgram;
  30796. CheckSource('TestAssert_SysUtils',
  30797. LinesToStr([ // statements
  30798. 'this.DoIt = function () {',
  30799. ' var b = false;',
  30800. ' var s = "";',
  30801. ' if (!b) throw pas.SysUtils.EAssertionFailed.$create("Create");',
  30802. ' if (!b) throw pas.SysUtils.EAssertionFailed.$create("Create$1", ["msg"]);',
  30803. '};',
  30804. '']),
  30805. LinesToStr([ // $mod.$main
  30806. '$mod.DoIt();',
  30807. '']));
  30808. end;
  30809. procedure TTestModule.TestObjectChecks;
  30810. begin
  30811. Scanner.CurrentBoolSwitches:=Scanner.CurrentBoolSwitches+[bsObjectChecks];
  30812. StartProgram(false);
  30813. Add([
  30814. 'type',
  30815. ' TObject = class',
  30816. ' procedure DoIt;',
  30817. ' end;',
  30818. ' TClass = class of tobject;',
  30819. ' TBird = class',
  30820. ' end;',
  30821. ' TBirdClass = class of TBird;',
  30822. 'var',
  30823. ' o : TObject;',
  30824. ' c: TClass;',
  30825. ' b: TBird;',
  30826. ' bc: TBirdClass;',
  30827. 'procedure TObject.DoIt;',
  30828. 'begin',
  30829. ' b:=TBird(o);',
  30830. 'end;',
  30831. 'begin',
  30832. ' o.DoIt;',
  30833. ' b:=TBird(o);',
  30834. ' bc:=TBirdClass(c);',
  30835. '']);
  30836. ConvertProgram;
  30837. CheckSource('TestCheckMethodCall',
  30838. LinesToStr([ // statements
  30839. 'rtl.createClass(this, "TObject", null, function () {',
  30840. ' this.$init = function () {',
  30841. ' };',
  30842. ' this.$final = function () {',
  30843. ' };',
  30844. ' this.DoIt = function () {',
  30845. ' rtl.checkMethodCall(this,$mod.TObject);',
  30846. ' $mod.b = rtl.asExt($mod.o, $mod.TBird, 1);',
  30847. ' };',
  30848. '});',
  30849. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  30850. '});',
  30851. 'this.o = null;',
  30852. 'this.c = null;',
  30853. 'this.b = null;',
  30854. 'this.bc = null;',
  30855. '']),
  30856. LinesToStr([ // $mod.$main
  30857. '$mod.o.DoIt();',
  30858. '$mod.b = rtl.asExt($mod.o,$mod.TBird, 1);',
  30859. '$mod.bc = rtl.asExt($mod.c, $mod.TBird, 2);',
  30860. '']));
  30861. end;
  30862. procedure TTestModule.TestOverflowChecks_Int;
  30863. begin
  30864. Scanner.CurrentBoolSwitches:=Scanner.CurrentBoolSwitches+[bsOverflowChecks];
  30865. StartProgram(false);
  30866. Add([
  30867. 'procedure DoIt;',
  30868. 'var',
  30869. ' b: byte;',
  30870. ' n: nativeint;',
  30871. ' u: nativeuint;',
  30872. ' c: currency;',
  30873. 'begin',
  30874. ' n:=n+n;',
  30875. ' n:=n-n;',
  30876. ' n:=n+b;',
  30877. ' n:=b-n;',
  30878. ' n:=n*n;',
  30879. ' n:=n*u;',
  30880. ' c:=c+b;',
  30881. ' c:=b+c;',
  30882. ' c:=c*b;',
  30883. ' c:=b*c;',
  30884. 'end;',
  30885. 'begin',
  30886. '']);
  30887. ConvertProgram;
  30888. CheckSource('TestOverflowChecks_Int',
  30889. LinesToStr([ // statements
  30890. 'this.DoIt = function () {',
  30891. ' var b = 0;',
  30892. ' var n = 0;',
  30893. ' var u = 0;',
  30894. ' var c = 0;',
  30895. ' n = rtl.oc(n + n);',
  30896. ' n = rtl.oc(n - n);',
  30897. ' n = rtl.oc(n + b);',
  30898. ' n = rtl.oc(b - n);',
  30899. ' n = rtl.oc(n * n);',
  30900. ' n = rtl.oc(n * u);',
  30901. ' c = rtl.oc(c + (b * 10000));',
  30902. ' c = rtl.oc((b * 10000) + c);',
  30903. ' c = rtl.oc(c * b);',
  30904. ' c = rtl.oc(b * c);',
  30905. '};',
  30906. '']),
  30907. LinesToStr([ // $mod.$main
  30908. '']));
  30909. end;
  30910. procedure TTestModule.TestRangeChecks_AssignInt;
  30911. begin
  30912. Scanner.Options:=Scanner.Options+[po_CAssignments];
  30913. StartProgram(false);
  30914. Add([
  30915. '{$R+}',
  30916. 'var',
  30917. ' b: byte = 2;',
  30918. ' w: word = 3;',
  30919. 'procedure DoIt(p: byte);',
  30920. 'begin',
  30921. ' b:=w;',
  30922. ' b+=w;',
  30923. ' b:=1;',
  30924. 'end;',
  30925. '{$R-}',
  30926. 'procedure DoSome;',
  30927. 'begin',
  30928. ' DoIt(w);',
  30929. ' b:=w;',
  30930. ' b:=2;',
  30931. 'end;',
  30932. 'begin',
  30933. '{$R+}',
  30934. '']);
  30935. ConvertProgram;
  30936. CheckSource('TestRangeChecks_AssignInt',
  30937. LinesToStr([ // statements
  30938. 'this.b = 2;',
  30939. 'this.w = 3;',
  30940. 'this.DoIt = function (p) {',
  30941. ' rtl.rc(p, 0, 255);',
  30942. ' $mod.b = rtl.rc($mod.w,0,255);',
  30943. ' rtl.rc($mod.b += $mod.w, 0, 255);',
  30944. ' $mod.b = 1;',
  30945. '};',
  30946. 'this.DoSome = function () {',
  30947. ' $mod.DoIt($mod.w);',
  30948. ' $mod.b = $mod.w;',
  30949. ' $mod.b = 2;',
  30950. '};',
  30951. '']),
  30952. LinesToStr([ // $mod.$main
  30953. '']));
  30954. end;
  30955. procedure TTestModule.TestRangeChecks_AssignIntRange;
  30956. begin
  30957. Scanner.Options:=Scanner.Options+[po_CAssignments];
  30958. StartProgram(false);
  30959. Add([
  30960. '{$R+}',
  30961. 'type Ten = 1..10;',
  30962. 'var',
  30963. ' b: Ten = 2;',
  30964. ' w: Ten = 3;',
  30965. 'procedure DoIt(p: Ten);',
  30966. 'begin',
  30967. ' b:=w;',
  30968. ' b+=w;',
  30969. ' b:=1;',
  30970. 'end;',
  30971. '{$R-}',
  30972. 'procedure DoSome;',
  30973. 'begin',
  30974. ' DoIt(w);',
  30975. ' b:=w;',
  30976. ' b:=2;',
  30977. 'end;',
  30978. 'begin',
  30979. '{$R+}',
  30980. '']);
  30981. ConvertProgram;
  30982. CheckSource('TestRangeChecks_AssignIntRange',
  30983. LinesToStr([ // statements
  30984. 'this.b = 2;',
  30985. 'this.w = 3;',
  30986. 'this.DoIt = function (p) {',
  30987. ' rtl.rc(p, 1, 10);',
  30988. ' $mod.b = rtl.rc($mod.w, 1, 10);',
  30989. ' rtl.rc($mod.b += $mod.w, 1, 10);',
  30990. ' $mod.b = 1;',
  30991. '};',
  30992. 'this.DoSome = function () {',
  30993. ' $mod.DoIt($mod.w);',
  30994. ' $mod.b = $mod.w;',
  30995. ' $mod.b = 2;',
  30996. '};',
  30997. '']),
  30998. LinesToStr([ // $mod.$main
  30999. '']));
  31000. end;
  31001. procedure TTestModule.TestRangeChecks_AssignEnum;
  31002. begin
  31003. StartProgram(false);
  31004. Add([
  31005. '{$R+}',
  31006. 'type TEnum = (red,green);',
  31007. 'var',
  31008. ' e: TEnum = red;',
  31009. 'procedure DoIt(p: TEnum);',
  31010. 'begin',
  31011. ' e:=p;',
  31012. ' p:=TEnum(0);',
  31013. ' p:=succ(e);',
  31014. 'end;',
  31015. '{$R-}',
  31016. 'procedure DoSome;',
  31017. 'begin',
  31018. ' DoIt(e);',
  31019. ' e:=TEnum(1);',
  31020. ' e:=pred(e);',
  31021. 'end;',
  31022. 'begin',
  31023. '{$R+}',
  31024. '']);
  31025. ConvertProgram;
  31026. CheckSource('TestRangeChecks_AssignEnum',
  31027. LinesToStr([ // statements
  31028. 'this.TEnum = {',
  31029. ' "0": "red",',
  31030. ' red: 0,',
  31031. ' "1": "green",',
  31032. ' green: 1',
  31033. '};',
  31034. 'this.e = this.TEnum.red;',
  31035. 'this.DoIt = function (p) {',
  31036. ' rtl.rc(p, 0, 1);',
  31037. ' $mod.e = rtl.rc(p, 0, 1);',
  31038. ' p = 0;',
  31039. ' p = rtl.rc($mod.e + 1, 0, 1);',
  31040. '};',
  31041. 'this.DoSome = function () {',
  31042. ' $mod.DoIt($mod.e);',
  31043. ' $mod.e = 1;',
  31044. ' $mod.e = $mod.e - 1;',
  31045. '};',
  31046. '']),
  31047. LinesToStr([ // $mod.$main
  31048. '']));
  31049. end;
  31050. procedure TTestModule.TestRangeChecks_AssignEnumRange;
  31051. begin
  31052. StartProgram(false);
  31053. Add([
  31054. '{$R+}',
  31055. 'type',
  31056. ' TEnum = (red,green);',
  31057. ' TEnumRg = red..green;',
  31058. 'var',
  31059. ' e: TEnumRg = red;',
  31060. 'procedure DoIt(p: TEnumRg);',
  31061. 'begin',
  31062. ' e:=p;',
  31063. ' p:=TEnumRg(0);',
  31064. ' p:=succ(e);',
  31065. 'end;',
  31066. '{$R-}',
  31067. 'procedure DoSome;',
  31068. 'begin',
  31069. ' DoIt(e);',
  31070. ' e:=TEnum(1);',
  31071. ' e:=pred(e);',
  31072. 'end;',
  31073. 'begin',
  31074. '{$R+}',
  31075. '']);
  31076. ConvertProgram;
  31077. CheckSource('TestRangeChecks_AssignEnumRange',
  31078. LinesToStr([ // statements
  31079. 'this.TEnum = {',
  31080. ' "0": "red",',
  31081. ' red: 0,',
  31082. ' "1": "green",',
  31083. ' green: 1',
  31084. '};',
  31085. 'this.e = this.TEnum.red;',
  31086. 'this.DoIt = function (p) {',
  31087. ' rtl.rc(p, 0, 1);',
  31088. ' $mod.e = rtl.rc(p, 0, 1);',
  31089. ' p = 0;',
  31090. ' p = rtl.rc($mod.e + 1, 0, 1);',
  31091. '};',
  31092. 'this.DoSome = function () {',
  31093. ' $mod.DoIt($mod.e);',
  31094. ' $mod.e = 1;',
  31095. ' $mod.e = $mod.e - 1;',
  31096. '};',
  31097. '']),
  31098. LinesToStr([ // $mod.$main
  31099. '']));
  31100. end;
  31101. procedure TTestModule.TestRangeChecks_AssignChar;
  31102. begin
  31103. StartProgram(false);
  31104. Add([
  31105. '{$R+}',
  31106. 'type',
  31107. ' TLetter = char;',
  31108. 'var',
  31109. ' b: TLetter = ''2'';',
  31110. ' w: TLetter = ''3'';',
  31111. 'procedure DoIt(p: TLetter);',
  31112. 'begin',
  31113. ' b:=w;',
  31114. ' b:=''1'';',
  31115. 'end;',
  31116. '{$R-}',
  31117. 'procedure DoSome;',
  31118. 'begin',
  31119. ' DoIt(w);',
  31120. ' b:=w;',
  31121. ' b:=''2'';',
  31122. 'end;',
  31123. 'begin',
  31124. '{$R+}',
  31125. '']);
  31126. ConvertProgram;
  31127. CheckSource('TestRangeChecks_AssignChar',
  31128. LinesToStr([ // statements
  31129. 'this.b = "2";',
  31130. 'this.w = "3";',
  31131. 'this.DoIt = function (p) {',
  31132. ' rtl.rcc(p, 0, 65535);',
  31133. ' $mod.b = rtl.rcc($mod.w, 0, 65535);',
  31134. ' $mod.b = "1";',
  31135. '};',
  31136. 'this.DoSome = function () {',
  31137. ' $mod.DoIt($mod.w);',
  31138. ' $mod.b = $mod.w;',
  31139. ' $mod.b = "2";',
  31140. '};',
  31141. '']),
  31142. LinesToStr([ // $mod.$main
  31143. '']));
  31144. end;
  31145. procedure TTestModule.TestRangeChecks_AssignCharRange;
  31146. begin
  31147. StartProgram(false);
  31148. Add([
  31149. '{$R+}',
  31150. 'type TDigit = ''0''..''9'';',
  31151. 'var',
  31152. ' b: TDigit = ''2'';',
  31153. ' w: TDigit = ''3'';',
  31154. 'procedure DoIt(p: TDigit);',
  31155. 'begin',
  31156. ' b:=w;',
  31157. ' b:=''1'';',
  31158. 'end;',
  31159. '{$R-}',
  31160. 'procedure DoSome;',
  31161. 'begin',
  31162. ' DoIt(w);',
  31163. ' b:=w;',
  31164. ' b:=''2'';',
  31165. 'end;',
  31166. 'begin',
  31167. '{$R+}',
  31168. '']);
  31169. ConvertProgram;
  31170. CheckSource('TestRangeChecks_AssignCharRange',
  31171. LinesToStr([ // statements
  31172. 'this.b = "2";',
  31173. 'this.w = "3";',
  31174. 'this.DoIt = function (p) {',
  31175. ' rtl.rcc(p, 48, 57);',
  31176. ' $mod.b = rtl.rcc($mod.w, 48, 57);',
  31177. ' $mod.b = "1";',
  31178. '};',
  31179. 'this.DoSome = function () {',
  31180. ' $mod.DoIt($mod.w);',
  31181. ' $mod.b = $mod.w;',
  31182. ' $mod.b = "2";',
  31183. '};',
  31184. '']),
  31185. LinesToStr([ // $mod.$main
  31186. '']));
  31187. end;
  31188. procedure TTestModule.TestRangeChecks_ArrayIndex;
  31189. begin
  31190. StartProgram(false);
  31191. Add([
  31192. '{$R+}',
  31193. 'type',
  31194. ' Ten = 1..10;',
  31195. ' TArr = array of Ten;',
  31196. ' TArrArr = array of TArr;',
  31197. ' TArrByte = array[byte] of Ten;',
  31198. ' TArrChar = array[''0''..''9''] of Ten;',
  31199. ' TArrByteChar = array[byte,''0''..''9''] of Ten;',
  31200. ' TObject = class',
  31201. ' A: TArr;',
  31202. ' end;',
  31203. 'procedure DoIt;',
  31204. 'var',
  31205. ' Arr: TArr;',
  31206. ' ArrArr: TArrArr;',
  31207. ' ArrByte: TArrByte;',
  31208. ' ArrChar: TArrChar;',
  31209. ' ArrByteChar: TArrByteChar;',
  31210. ' i: Ten;',
  31211. ' c: char;',
  31212. ' o: tobject;',
  31213. 'begin',
  31214. ' i:=Arr[1];',
  31215. ' i:=ArrByteChar[1,''2''];',
  31216. ' Arr[1]:=Arr[1];',
  31217. ' Arr[i]:=Arr[i];',
  31218. ' ArrByte[3]:=ArrByte[3];',
  31219. ' ArrByte[i]:=ArrByte[i];',
  31220. ' ArrChar[''5'']:=ArrChar[''5''];',
  31221. ' ArrChar[c]:=ArrChar[c];',
  31222. ' ArrByteChar[7,''7'']:=ArrByteChar[7,''7''];',
  31223. ' ArrByteChar[i,c]:=ArrByteChar[i,c];',
  31224. ' o.a[i]:=o.a[i];',
  31225. 'end;',
  31226. 'begin',
  31227. '']);
  31228. ConvertProgram;
  31229. CheckSource('TestRangeChecks_ArrayIndex',
  31230. LinesToStr([ // statements
  31231. 'rtl.createClass(this, "TObject", null, function () {',
  31232. ' this.$init = function () {',
  31233. ' this.A = [];',
  31234. ' };',
  31235. ' this.$final = function () {',
  31236. ' this.A = undefined;',
  31237. ' };',
  31238. '});',
  31239. 'this.DoIt = function () {',
  31240. ' var Arr = [];',
  31241. ' var ArrArr = [];',
  31242. ' var ArrByte = rtl.arraySetLength(null, 0, 256);',
  31243. ' var ArrChar = rtl.arraySetLength(null, 0, 10);',
  31244. ' var ArrByteChar = rtl.arraySetLength(null, 0, 256, 10);',
  31245. ' var i = 0;',
  31246. ' var c = "";',
  31247. ' var o = null;',
  31248. ' i = rtl.rc(Arr[1], 1, 10);',
  31249. ' i = rtl.rc(ArrByteChar[1][2], 1, 10);',
  31250. ' Arr[1] = rtl.rc(Arr[1], 1, 10);',
  31251. ' rtl.rcArrW(Arr, i, rtl.rcArrR(Arr, i));',
  31252. ' ArrByte[3] = rtl.rc(ArrByte[3], 1, 10);',
  31253. ' rtl.rcArrW(ArrByte, i, rtl.rcArrR(ArrByte, i));',
  31254. ' ArrChar[5] = rtl.rc(ArrChar[5], 1, 10);',
  31255. ' rtl.rcArrW(ArrChar, c.charCodeAt() - 48, rtl.rcArrR(ArrChar, c.charCodeAt() - 48));',
  31256. ' ArrByteChar[7][7] = rtl.rc(ArrByteChar[7][7], 1, 10);',
  31257. ' rtl.rcArrW(ArrByteChar, i, c.charCodeAt() - 48, rtl.rcArrR(ArrByteChar, i, c.charCodeAt() - 48));',
  31258. ' rtl.rcArrW(o.A, i, rtl.rcArrR(o.A, i));',
  31259. '};',
  31260. '']),
  31261. LinesToStr([ // $mod.$main
  31262. '']));
  31263. end;
  31264. procedure TTestModule.TestRangeChecks_ArrayOfRecIndex;
  31265. begin
  31266. StartProgram(false);
  31267. Add([
  31268. '{$R+}',
  31269. 'type',
  31270. ' Ten = 1..10;',
  31271. ' TRec = record x: Ten end;',
  31272. ' TArr = array of TRec;',
  31273. ' TArrArr = array of TArr;',
  31274. ' TObject = class',
  31275. ' A: TArr;',
  31276. ' end;',
  31277. 'procedure DoIt;',
  31278. 'var',
  31279. ' Arr: TArr;',
  31280. ' ArrArr: TArrArr;',
  31281. ' i: Ten;',
  31282. ' o: tobject;',
  31283. 'begin',
  31284. ' Arr[1]:=Arr[1];',
  31285. ' Arr[i]:=Arr[i+1];',
  31286. ' o.a[i]:=o.a[i+2];',
  31287. 'end;',
  31288. 'begin',
  31289. '']);
  31290. ConvertProgram;
  31291. CheckSource('TestRangeChecks_ArrayOfRecIndex',
  31292. LinesToStr([ // statements
  31293. 'rtl.recNewT(this, "TRec", function () {',
  31294. ' this.x = 0;',
  31295. ' this.$eq = function (b) {',
  31296. ' return this.x === b.x;',
  31297. ' };',
  31298. ' this.$assign = function (s) {',
  31299. ' this.x = s.x;',
  31300. ' return this;',
  31301. ' };',
  31302. '});',
  31303. 'rtl.createClass(this, "TObject", null, function () {',
  31304. ' this.$init = function () {',
  31305. ' this.A = [];',
  31306. ' };',
  31307. ' this.$final = function () {',
  31308. ' this.A = undefined;',
  31309. ' };',
  31310. '});',
  31311. 'this.DoIt = function () {',
  31312. ' var Arr = [];',
  31313. ' var ArrArr = [];',
  31314. ' var i = 0;',
  31315. ' var o = null;',
  31316. ' Arr[1].$assign(Arr[1]);',
  31317. ' rtl.rcArrR(Arr, i).$assign(rtl.rcArrR(Arr, i + 1));',
  31318. ' rtl.rcArrR(o.A, i).$assign(rtl.rcArrR(o.A, i + 2));',
  31319. '};',
  31320. '']),
  31321. LinesToStr([ // $mod.$main
  31322. '']));
  31323. end;
  31324. procedure TTestModule.TestRangeChecks_StringIndex;
  31325. begin
  31326. StartProgram(false);
  31327. Add([
  31328. 'type',
  31329. ' TObject = class',
  31330. ' S: string;',
  31331. ' end;',
  31332. '{$R+}',
  31333. 'procedure DoIt(var h: string);',
  31334. 'var',
  31335. ' s: string;',
  31336. ' i: longint;',
  31337. ' c: char;',
  31338. ' o: tobject;',
  31339. 'begin',
  31340. ' c:=s[1];',
  31341. ' s[i]:=s[i];',
  31342. ' h[i]:=h[i];',
  31343. ' c:=o.s[i];',
  31344. ' o.s[i]:=c;',
  31345. 'end;',
  31346. 'begin',
  31347. '']);
  31348. ConvertProgram;
  31349. CheckSource('TestRangeChecks_StringIndex',
  31350. LinesToStr([ // statements
  31351. 'rtl.createClass(this, "TObject", null, function () {',
  31352. ' this.$init = function () {',
  31353. ' this.S = "";',
  31354. ' };',
  31355. ' this.$final = function () {',
  31356. ' };',
  31357. '});',
  31358. 'this.DoIt = function (h) {',
  31359. ' var s = "";',
  31360. ' var i = 0;',
  31361. ' var c = "";',
  31362. ' var o = null;',
  31363. ' c = rtl.rcc(rtl.rcCharAt(s, 0), 0, 65535);',
  31364. ' s = rtl.rcSetCharAt(s, i - 1, rtl.rcCharAt(s, i - 1));',
  31365. ' h.set(rtl.rcSetCharAt(h.get(), i - 1, rtl.rcCharAt(h.get(), i - 1)));',
  31366. ' c = rtl.rcc(rtl.rcCharAt(o.S, i - 1), 0, 65535);',
  31367. ' o.S = rtl.rcSetCharAt(o.S, i - 1, c);',
  31368. '};',
  31369. '']),
  31370. LinesToStr([ // $mod.$main
  31371. '']));
  31372. end;
  31373. procedure TTestModule.TestRangeChecks_TypecastInt;
  31374. begin
  31375. StartProgram(false);
  31376. Add([
  31377. '{$R+}',
  31378. 'var',
  31379. ' i: nativeint;',
  31380. ' b: byte;',
  31381. ' sh: shortint;',
  31382. ' w: word;',
  31383. ' sm: smallint;',
  31384. ' lw: longword;',
  31385. ' li: longint;',
  31386. 'begin',
  31387. ' b:=12+byte(i);',
  31388. ' sh:=12+shortint(i);',
  31389. ' w:=12+word(i);',
  31390. ' sm:=12+smallint(i);',
  31391. ' lw:=12+longword(i);',
  31392. ' li:=12+longint(i);',
  31393. '']);
  31394. ConvertProgram;
  31395. CheckSource('TestRangeChecks_TypecastInt',
  31396. LinesToStr([
  31397. 'this.i = 0;',
  31398. 'this.b = 0;',
  31399. 'this.sh = 0;',
  31400. 'this.w = 0;',
  31401. 'this.sm = 0;',
  31402. 'this.lw = 0;',
  31403. 'this.li = 0;',
  31404. '']),
  31405. LinesToStr([
  31406. '$mod.b = rtl.rc(12 + rtl.rc($mod.i, 0, 255), 0, 255);',
  31407. '$mod.sh = rtl.rc(12 + rtl.rc($mod.i, -128, 127), -128, 127);',
  31408. '$mod.w = rtl.rc(12 + rtl.rc($mod.i, 0, 65535), 0, 65535);',
  31409. '$mod.sm = rtl.rc(12 + rtl.rc($mod.i, -32768, 32767), -32768, 32767);',
  31410. '$mod.lw = rtl.rc(12 + rtl.rc($mod.i, 0, 4294967295), 0, 4294967295);',
  31411. '$mod.li = rtl.rc(12 + rtl.rc($mod.i, -2147483648, 2147483647), -2147483648, 2147483647);',
  31412. '']));
  31413. end;
  31414. procedure TTestModule.TestRangeChecks_TypeHelperInt;
  31415. begin
  31416. Scanner.Options:=Scanner.Options+[po_CAssignments];
  31417. StartProgram(false);
  31418. Add([
  31419. '{$modeswitch typehelpers}',
  31420. '{$R+}',
  31421. 'type',
  31422. ' TObject = class',
  31423. ' FSize: byte;',
  31424. ' property Size: byte read FSize;',
  31425. ' end;',
  31426. ' THelper = type helper for byte',
  31427. ' procedure SetIt(w: word);',
  31428. ' end;',
  31429. 'procedure THelper.SetIt(w: word);',
  31430. 'begin',
  31431. ' Self:=w;',
  31432. 'end;',
  31433. 'function GetIt: byte;',
  31434. 'begin',
  31435. ' Result.SetIt(2);',
  31436. 'end;',
  31437. 'var',
  31438. ' b: byte = 3;',
  31439. ' o: TObject;',
  31440. 'begin',
  31441. ' b.SetIt(14);',
  31442. ' with b do SetIt(15);',
  31443. ' o.Size.SetIt(16);',
  31444. '']);
  31445. ConvertProgram;
  31446. CheckSource('TestRangeChecks_AssignInt',
  31447. LinesToStr([ // statements
  31448. 'rtl.createClass(this, "TObject", null, function () {',
  31449. ' this.$init = function () {',
  31450. ' this.FSize = 0;',
  31451. ' };',
  31452. ' this.$final = function () {',
  31453. ' };',
  31454. '});',
  31455. 'rtl.createHelper(this, "THelper", null, function () {',
  31456. ' this.SetIt = function (w) {',
  31457. ' rtl.rc(w, 0, 65535);',
  31458. ' this.set(w);',
  31459. ' };',
  31460. '});',
  31461. 'this.GetIt = function () {',
  31462. ' var Result = 0;',
  31463. ' $mod.THelper.SetIt.call({',
  31464. ' get: function () {',
  31465. ' return Result;',
  31466. ' },',
  31467. ' set: function (v) {',
  31468. ' rtl.rc(v, 0, 255);',
  31469. ' Result = v;',
  31470. ' }',
  31471. ' }, 2);',
  31472. ' return Result;',
  31473. '};',
  31474. 'this.b = 3;',
  31475. 'this.o = null;',
  31476. '']),
  31477. LinesToStr([ // $mod.$main
  31478. '$mod.THelper.SetIt.call({',
  31479. ' p: $mod,',
  31480. ' get: function () {',
  31481. ' return this.p.b;',
  31482. ' },',
  31483. ' set: function (v) {',
  31484. ' rtl.rc(v, 0, 255);',
  31485. ' this.p.b = v;',
  31486. ' }',
  31487. '}, 14);',
  31488. 'var $with = $mod.b;',
  31489. '$mod.THelper.SetIt.call({',
  31490. ' get: function () {',
  31491. ' return $with;',
  31492. ' },',
  31493. ' set: function (v) {',
  31494. ' rtl.rc(v, 0, 255);',
  31495. ' $with = v;',
  31496. ' }',
  31497. '}, 15);',
  31498. '$mod.THelper.SetIt.call({',
  31499. ' p: $mod.o,',
  31500. ' get: function () {',
  31501. ' return this.p.FSize;',
  31502. ' },',
  31503. ' set: function (v) {',
  31504. ' rtl.rc(v, 0, 255);',
  31505. ' this.p.FSize = v;',
  31506. ' }',
  31507. '}, 16);',
  31508. '']));
  31509. end;
  31510. procedure TTestModule.TestAsync_Proc;
  31511. begin
  31512. StartProgram(false);
  31513. Add([
  31514. 'procedure Fly(w: word = 1); async; forward;',
  31515. 'procedure Run(w: word = 2); async;',
  31516. 'begin',
  31517. ' Fly(w);',
  31518. ' Fly;',
  31519. ' await(Fly(w));',
  31520. ' await(Fly);',
  31521. 'end;',
  31522. 'procedure Fly(w: word); ',
  31523. 'begin',
  31524. 'end;',
  31525. 'begin',
  31526. ' Run;',
  31527. ' Run(3);',
  31528. '']);
  31529. CheckResolverUnexpectedHints();
  31530. ConvertProgram;
  31531. CheckSource('TestAsync_Proc',
  31532. LinesToStr([ // statements
  31533. 'this.Run = async function (w) {',
  31534. ' $mod.Fly(w);',
  31535. ' $mod.Fly(1);',
  31536. ' await $mod.Fly(w);',
  31537. ' await $mod.Fly(1);',
  31538. '};',
  31539. 'this.Fly = async function (w) {',
  31540. '};',
  31541. '']),
  31542. LinesToStr([
  31543. '$mod.Run(2);',
  31544. '$mod.Run(3);',
  31545. '']));
  31546. end;
  31547. procedure TTestModule.TestAsync_CallResultIsPromise;
  31548. begin
  31549. StartProgram(false);
  31550. Add([
  31551. '{$modeswitch externalclass}',
  31552. 'type',
  31553. ' TObject = class',
  31554. ' end;',
  31555. ' TJSPromise = class external name ''Promise''',
  31556. ' end;',
  31557. ' TBird = class',
  31558. ' function Fly: word; async; ',
  31559. ' end;',
  31560. 'function TBird.Fly: word; async; ',
  31561. 'begin',
  31562. ' Result:=3;',
  31563. ' Fly:=4+Result;',
  31564. ' if Result=5 then ;',
  31565. ' exit(6);',
  31566. 'end;',
  31567. 'function Run: word; async;',
  31568. 'begin',
  31569. ' Result:=11+Result;',
  31570. ' inc(Result);',
  31571. 'end;',
  31572. 'var',
  31573. ' p: TJSPromise;',
  31574. ' o: TBird;',
  31575. 'begin',
  31576. ' p:=Run;',
  31577. ' p:=Run();',
  31578. ' if Run=p then ;',
  31579. ' if p=Run then ;',
  31580. ' if Run()=p then ;',
  31581. ' if p=Run() then ;',
  31582. ' p:=o.Fly;',
  31583. ' p:=o.Fly();',
  31584. ' if o.Fly=p then ;',
  31585. ' if o.Fly()=p then ;',
  31586. ' with o do begin',
  31587. ' p:=Fly;',
  31588. ' p:=Fly();',
  31589. ' if Fly=p then ;',
  31590. ' if Fly()=p then ;',
  31591. ' end;',
  31592. '']);
  31593. CheckResolverUnexpectedHints();
  31594. ConvertProgram;
  31595. CheckSource('TestAsync_CallResultIsPromise',
  31596. LinesToStr([ // statements
  31597. 'rtl.createClass(this, "TObject", null, function () {',
  31598. ' this.$init = function () {',
  31599. ' };',
  31600. ' this.$final = function () {',
  31601. ' };',
  31602. '});',
  31603. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  31604. ' this.Fly = async function () {',
  31605. ' var Result = 0;',
  31606. ' Result = 3;',
  31607. ' Result = 4 + Result;',
  31608. ' if (Result === 5) ;',
  31609. ' return 6;',
  31610. ' return Result;',
  31611. ' };',
  31612. '});',
  31613. 'this.Run = async function () {',
  31614. ' var Result = 0;',
  31615. ' Result = 11 + Result;',
  31616. ' Result += 1;',
  31617. ' return Result;',
  31618. '};',
  31619. 'this.p = null;',
  31620. 'this.o = null;',
  31621. '']),
  31622. LinesToStr([
  31623. '$mod.p = $mod.Run();',
  31624. '$mod.p = $mod.Run();',
  31625. 'if ($mod.Run() === $mod.p) ;',
  31626. 'if ($mod.p === $mod.Run()) ;',
  31627. 'if ($mod.Run() === $mod.p) ;',
  31628. 'if ($mod.p === $mod.Run()) ;',
  31629. '$mod.p = $mod.o.Fly();',
  31630. '$mod.p = $mod.o.Fly();',
  31631. 'if ($mod.o.Fly() === $mod.p) ;',
  31632. 'if ($mod.o.Fly() === $mod.p) ;',
  31633. 'var $with = $mod.o;',
  31634. '$mod.p = $with.Fly();',
  31635. '$mod.p = $with.Fly();',
  31636. 'if ($with.Fly() === $mod.p) ;',
  31637. 'if ($with.Fly() === $mod.p) ;',
  31638. '']));
  31639. end;
  31640. procedure TTestModule.TestAsync_ConstructorFail;
  31641. begin
  31642. StartProgram(false);
  31643. Add([
  31644. 'type',
  31645. ' TObject = class',
  31646. ' end;',
  31647. ' TBird = class',
  31648. ' constructor Create; async;',
  31649. ' end;',
  31650. 'constructor TBird.Create; async;',
  31651. 'begin',
  31652. 'end;',
  31653. 'begin',
  31654. '']);
  31655. SetExpectedPasResolverError('Invalid constructor modifier async',nInvalidXModifierY);
  31656. ConvertProgram;
  31657. end;
  31658. procedure TTestModule.TestAsync_PropertyGetterFail;
  31659. begin
  31660. StartProgram(false);
  31661. Add([
  31662. 'type',
  31663. ' TObject = class',
  31664. ' end;',
  31665. ' TBird = class',
  31666. ' function GetSize: word; async;',
  31667. ' property Size: word read GetSize;',
  31668. ' end;',
  31669. 'function TBird.GetSize: word; async;',
  31670. 'begin',
  31671. 'end;',
  31672. 'begin',
  31673. '']);
  31674. SetExpectedPasResolverError('Invalid property getter modifier async',nInvalidXModifierY);
  31675. ConvertProgram;
  31676. end;
  31677. procedure TTestModule.TestAwait_NonPromiseWithTypeFail;
  31678. begin
  31679. StartProgram(false);
  31680. Add([
  31681. 'procedure Run; async;',
  31682. 'begin',
  31683. ' await(word,1);',
  31684. 'end;',
  31685. 'begin',
  31686. '']);
  31687. SetExpectedPasResolverError('Incompatible type arg no. 2: Got "Longint", expected "TJSPromise"',nIncompatibleTypeArgNo);
  31688. ConvertProgram;
  31689. end;
  31690. procedure TTestModule.TestAwait_AsyncCallTypeMismatch;
  31691. begin
  31692. StartProgram(false);
  31693. Add([
  31694. 'type',
  31695. ' TObject = class',
  31696. ' end;',
  31697. ' TBird = class',
  31698. ' end;',
  31699. 'function Fly: TObject; async;',
  31700. 'begin',
  31701. 'end;',
  31702. 'procedure Run; async;',
  31703. 'begin',
  31704. ' await(TBird,Fly);',
  31705. 'end;',
  31706. 'begin',
  31707. '']);
  31708. SetExpectedPasResolverError('Incompatible type arg no. 2: Got "TObject", expected "TBird"',nIncompatibleTypeArgNo);
  31709. ConvertProgram;
  31710. end;
  31711. procedure TTestModule.TestAWait_OutsideAsyncFail;
  31712. begin
  31713. StartProgram(false);
  31714. Add([
  31715. 'procedure Crawl(w: double); ',
  31716. 'begin',
  31717. 'end;',
  31718. 'procedure Run(w: double);',
  31719. 'begin',
  31720. ' await(Crawl(w));',
  31721. 'end;',
  31722. 'begin',
  31723. ' Run(1);']);
  31724. SetExpectedPasResolverError(sAWaitOnlyInAsyncProcedure,nAWaitOnlyInAsyncProcedure);
  31725. ConvertProgram;
  31726. end;
  31727. procedure TTestModule.TestAWait_Result;
  31728. begin
  31729. StartProgram(false);
  31730. Add([
  31731. '{$modeswitch externalclass}',
  31732. 'type',
  31733. ' TJSPromise = class external name ''Promise''',
  31734. ' end;',
  31735. 'function Crawl(d: double = 1.3): word; ',
  31736. 'begin',
  31737. 'end;',
  31738. 'function Run(d: double = 1.6): word; async;',
  31739. 'begin',
  31740. ' Result:=await(1);',
  31741. ' Result:=await(Crawl);',
  31742. ' Result:=await(Crawl(4.5));',
  31743. ' Result:=await(Run);',
  31744. ' Result:=await(Run(6.7));',
  31745. 'end;',
  31746. 'begin',
  31747. ' Run(1);']);
  31748. ConvertProgram;
  31749. CheckSource('TestAWait_Result',
  31750. LinesToStr([ // statements
  31751. 'this.Crawl = function (d) {',
  31752. ' var Result = 0;',
  31753. ' return Result;',
  31754. '};',
  31755. 'this.Run = async function (d) {',
  31756. ' var Result = 0;',
  31757. ' Result = await 1;',
  31758. ' Result = await $mod.Crawl(1.3);',
  31759. ' Result = await $mod.Crawl(4.5);',
  31760. ' Result = await $mod.Run(1.6);',
  31761. ' Result = await $mod.Run(6.7);',
  31762. ' return Result;',
  31763. '};',
  31764. '']),
  31765. LinesToStr([
  31766. '$mod.Run(1);'
  31767. ]));
  31768. SetExpectedPasResolverError('Await without promise',nAwaitWithoutPromise);
  31769. end;
  31770. procedure TTestModule.TestAWait_ExternalClassPromise;
  31771. begin
  31772. StartProgram(false);
  31773. Add([
  31774. '{$modeswitch externalclass}',
  31775. 'type',
  31776. ' TJSPromise = class external name ''Promise''',
  31777. ' end;',
  31778. 'function Fly(w: word): TJSPromise;',
  31779. 'begin',
  31780. 'end;',
  31781. 'function Jump(w: word): word; async;',
  31782. 'begin',
  31783. 'end;',
  31784. 'function Eat(w: word): TJSPromise; async;',
  31785. 'begin',
  31786. 'end;',
  31787. 'function Run(d: double): word; async;',
  31788. 'var',
  31789. ' p: TJSPromise;',
  31790. 'begin',
  31791. ' Result:=await(word,p);', // promise needs type
  31792. ' Result:=await(word,Fly(3));', // promise needs type
  31793. ' Result:=await(Jump(4));', // async non promise must omit the type
  31794. ' Result:=await(word,Jump(5));', // async call can provide fitting type
  31795. ' Result:=await(word,Eat(6));', // promise needs type
  31796. 'end;',
  31797. 'begin',
  31798. '']);
  31799. ConvertProgram;
  31800. CheckSource('TestAWait_ExternalClassPromise',
  31801. LinesToStr([ // statements
  31802. 'this.Fly = function (w) {',
  31803. ' var Result = null;',
  31804. ' return Result;',
  31805. '};',
  31806. 'this.Jump = async function (w) {',
  31807. ' var Result = 0;',
  31808. ' return Result;',
  31809. '};',
  31810. 'this.Eat = async function (w) {',
  31811. ' var Result = null;',
  31812. ' return Result;',
  31813. '};',
  31814. 'this.Run = async function (d) {',
  31815. ' var Result = 0;',
  31816. ' var p = null;',
  31817. ' Result = await p;',
  31818. ' Result = await $mod.Fly(3);',
  31819. ' Result = await $mod.Jump(4);',
  31820. ' Result = await $mod.Jump(5);',
  31821. ' Result = await $mod.Eat(6);',
  31822. ' return Result;',
  31823. '};',
  31824. '']),
  31825. LinesToStr([
  31826. ]));
  31827. CheckResolverUnexpectedHints();
  31828. end;
  31829. procedure TTestModule.TestAsync_AnonymousProc;
  31830. begin
  31831. StartProgram(false);
  31832. Add([
  31833. '{$mode objfpc}',
  31834. '{$modeswitch externalclass}',
  31835. 'type',
  31836. ' TJSPromise = class external name ''Promise''',
  31837. ' end;',
  31838. 'type',
  31839. ' TFunc = reference to function(x: double): word; async;',
  31840. 'function Crawl(d: double = 1.3): word; async;',
  31841. 'begin',
  31842. 'end;',
  31843. 'var Func: TFunc;',
  31844. 'begin',
  31845. ' Func:=function(c:double):word async begin',
  31846. ' Result:=await(Crawl(c));',
  31847. ' end;',
  31848. ' Func:=function(c:double):word async assembler asm',
  31849. ' end;',
  31850. ' ']);
  31851. ConvertProgram;
  31852. CheckSource('TestAsync_AnonymousProc',
  31853. LinesToStr([ // statements
  31854. 'this.Crawl = async function (d) {',
  31855. ' var Result = 0;',
  31856. ' return Result;',
  31857. '};',
  31858. 'this.Func = null;',
  31859. '']),
  31860. LinesToStr([
  31861. '$mod.Func = async function (c) {',
  31862. ' var Result = 0;',
  31863. ' Result = await $mod.Crawl(c);',
  31864. ' return Result;',
  31865. '};',
  31866. '$mod.Func = async function (c) {',
  31867. '};',
  31868. '']));
  31869. CheckResolverUnexpectedHints();
  31870. end;
  31871. procedure TTestModule.TestAsync_ProcType;
  31872. begin
  31873. StartProgram(false);
  31874. Add([
  31875. '{$mode objfpc}',
  31876. 'type',
  31877. ' TRefFunc = reference to function(x: double = 1.3): word; async;',
  31878. ' TFunc = function(x: double = 1.1): word; async;',
  31879. ' TProc = procedure(x: longint = 7); async;',
  31880. 'function Crawl(d: double): word; async;',
  31881. 'begin',
  31882. 'end;',
  31883. 'procedure Run(e:longint); async;',
  31884. 'begin',
  31885. 'end;',
  31886. 'procedure Fly(p: TProc); async;',
  31887. 'begin',
  31888. ' await(p);',
  31889. ' await(p());',
  31890. 'end;',
  31891. 'var',
  31892. ' RefFunc: TRefFunc;',
  31893. ' Func: TFunc;',
  31894. ' Proc, ProcB: TProc;',
  31895. 'begin',
  31896. ' Func:=@Crawl;',
  31897. ' RefFunc:=@Crawl;',
  31898. ' RefFunc:=function(c:double):word async begin',
  31899. ' Result:=await(RefFunc);',
  31900. ' Result:=await(RefFunc());',
  31901. ' Result:=await(Func);',
  31902. ' Result:=await(Func());',
  31903. ' await(Proc);',
  31904. ' await(Proc());',
  31905. ' await(Proc(13));',
  31906. ' end;',
  31907. ' Proc:=@Run;',
  31908. ' if Proc=ProcB then ;',
  31909. ' ']);
  31910. ConvertProgram;
  31911. CheckResolverUnexpectedHints();
  31912. CheckSource('TestAsync_ProcType',
  31913. LinesToStr([ // statements
  31914. 'this.Crawl = async function (d) {',
  31915. ' var Result = 0;',
  31916. ' return Result;',
  31917. '};',
  31918. 'this.Run = async function (e) {',
  31919. '};',
  31920. 'this.Fly = async function (p) {',
  31921. ' await p(7);',
  31922. ' await p(7);',
  31923. '};',
  31924. 'this.RefFunc = null;',
  31925. 'this.Func = null;',
  31926. 'this.Proc = null;',
  31927. 'this.ProcB = null;',
  31928. '']),
  31929. LinesToStr([
  31930. '$mod.Func = $mod.Crawl;',
  31931. '$mod.RefFunc = $mod.Crawl;',
  31932. '$mod.RefFunc = async function (c) {',
  31933. ' var Result = 0;',
  31934. ' Result = await $mod.RefFunc(1.3);',
  31935. ' Result = await $mod.RefFunc(1.3);',
  31936. ' Result = await $mod.Func(1.1);',
  31937. ' Result = await $mod.Func(1.1);',
  31938. ' await $mod.Proc(7);',
  31939. ' await $mod.Proc(7);',
  31940. ' await $mod.Proc(13);',
  31941. ' return Result;',
  31942. '};',
  31943. '$mod.Proc = $mod.Run;',
  31944. 'if (rtl.eqCallback($mod.Proc, $mod.ProcB)) ;',
  31945. '']));
  31946. end;
  31947. procedure TTestModule.TestAsync_ProcTypeAsyncModMismatchFail;
  31948. begin
  31949. StartProgram(false);
  31950. Add([
  31951. '{$mode objfpc}',
  31952. 'type',
  31953. ' TRefFunc = reference to function(x: double = 1.3): word;',
  31954. 'function Crawl(d: double): word; async;',
  31955. 'begin',
  31956. 'end;',
  31957. 'var',
  31958. ' RefFunc: TRefFunc;',
  31959. 'begin',
  31960. ' RefFunc:=@Crawl;',
  31961. ' ']);
  31962. SetExpectedPasResolverError('procedure type modifier "async" mismatch',nXModifierMismatchY);
  31963. ConvertProgram;
  31964. end;
  31965. procedure TTestModule.TestAsync_Inherited;
  31966. begin
  31967. StartProgram(false);
  31968. Add([
  31969. '{$mode objfpc}',
  31970. '{$modeswitch externalclass}',
  31971. 'type',
  31972. ' TJSPromise = class external name ''Promise''',
  31973. ' end;',
  31974. ' TObject = class',
  31975. ' function Run(w: word = 3): word; async; virtual;',
  31976. ' end;',
  31977. ' TBird = class',
  31978. ' function Run(w: word = 3): word; async; override;',
  31979. ' end;',
  31980. 'function TObject.Run(w: word = 3): word; async;',
  31981. 'begin',
  31982. 'end;',
  31983. 'function TBird.Run(w: word = 3): word;', // async modifier not needed in impl
  31984. 'var p: TJSPromise;',
  31985. 'begin',
  31986. ' p:=inherited;',
  31987. ' p:=inherited Run;',
  31988. ' p:=inherited Run();',
  31989. ' p:=inherited Run(4);',
  31990. ' exit(p);',
  31991. ' exit(inherited);',
  31992. ' exit(inherited Run);',
  31993. ' exit(inherited Run(5));',
  31994. ' exit(6);',
  31995. 'end;',
  31996. 'begin',
  31997. ' ']);
  31998. ConvertProgram;
  31999. CheckSource('TestAsync_Inherited',
  32000. LinesToStr([ // statements
  32001. 'rtl.createClass(this, "TObject", null, function () {',
  32002. ' this.$init = function () {',
  32003. ' };',
  32004. ' this.$final = function () {',
  32005. ' };',
  32006. ' this.Run = async function (w) {',
  32007. ' var Result = 0;',
  32008. ' return Result;',
  32009. ' };',
  32010. '});',
  32011. 'rtl.createClass(this, "TBird", this.TObject, function () {',
  32012. ' this.Run = async function (w) {',
  32013. ' var Result = 0;',
  32014. ' var p = null;',
  32015. ' p = $mod.TObject.Run.apply(this, arguments);',
  32016. ' p = $mod.TObject.Run.call(this, 3);',
  32017. ' p = $mod.TObject.Run.call(this, 3);',
  32018. ' p = $mod.TObject.Run.call(this, 4);',
  32019. ' return p;',
  32020. ' return $mod.TObject.Run.apply(this, arguments);',
  32021. ' return $mod.TObject.Run.call(this, 3);',
  32022. ' return $mod.TObject.Run.call(this, 5);',
  32023. ' return 6;',
  32024. ' return Result;',
  32025. ' };',
  32026. '});',
  32027. '']),
  32028. LinesToStr([
  32029. '']));
  32030. CheckResolverUnexpectedHints();
  32031. end;
  32032. procedure TTestModule.TestAsync_ClassInterface;
  32033. begin
  32034. StartProgram(false);
  32035. Add([
  32036. '{$mode objfpc}',
  32037. '{$modeswitch externalclass}',
  32038. 'type',
  32039. ' TJSPromise = class external name ''Promise''',
  32040. ' end;',
  32041. ' IUnknown = interface',
  32042. ' function _AddRef: longint;',
  32043. ' function _Release: longint;',
  32044. ' end;',
  32045. 'function Run: IUnknown; async;',
  32046. 'begin',
  32047. 'end;',
  32048. 'procedure Fly;',
  32049. 'var p: TJSPromise;',
  32050. 'begin',
  32051. ' Run;',
  32052. ' Run();',
  32053. ' p:=Run;',
  32054. ' p:=Run();',
  32055. 'end;',
  32056. 'begin',
  32057. ' ']);
  32058. ConvertProgram;
  32059. CheckSource('TestAsync_ClassInterface',
  32060. LinesToStr([ // statements
  32061. 'rtl.createInterface(this, "IUnknown", "{D7ADB0E1-758A-322B-BDDF-21CD521DDFA9}", ["_AddRef", "_Release"], null);',
  32062. 'this.Run = async function () {',
  32063. ' var Result = null;',
  32064. ' return Result;',
  32065. '};',
  32066. 'this.Fly = function () {',
  32067. ' var p = null;',
  32068. ' $mod.Run();',
  32069. ' $mod.Run();',
  32070. ' p = $mod.Run();',
  32071. ' p = $mod.Run();',
  32072. '};',
  32073. '']),
  32074. LinesToStr([
  32075. '']));
  32076. CheckResolverUnexpectedHints();
  32077. end;
  32078. Initialization
  32079. RegisterTests([TTestModule]);
  32080. end.