| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662466346644665466646674668466946704671467246734674467546764677467846794680468146824683468446854686468746884689469046914692469346944695469646974698469947004701470247034704470547064707470847094710471147124713471447154716471747184719472047214722472347244725472647274728472947304731473247334734473547364737473847394740474147424743474447454746474747484749475047514752475347544755475647574758475947604761476247634764476547664767476847694770477147724773477447754776477747784779478047814782478347844785478647874788478947904791479247934794479547964797479847994800480148024803480448054806480748084809481048114812481348144815481648174818481948204821482248234824482548264827482848294830483148324833483448354836483748384839484048414842484348444845484648474848484948504851485248534854485548564857485848594860486148624863486448654866486748684869487048714872487348744875487648774878487948804881488248834884488548864887488848894890489148924893489448954896489748984899490049014902490349044905490649074908490949104911491249134914491549164917491849194920492149224923492449254926492749284929493049314932493349344935493649374938493949404941494249434944494549464947494849494950495149524953495449554956495749584959496049614962496349644965496649674968496949704971497249734974497549764977497849794980498149824983498449854986498749884989499049914992499349944995499649974998499950005001500250035004500550065007500850095010501150125013501450155016501750185019502050215022502350245025502650275028502950305031503250335034503550365037503850395040504150425043504450455046504750485049505050515052505350545055505650575058505950605061506250635064506550665067506850695070507150725073507450755076507750785079508050815082508350845085508650875088508950905091509250935094509550965097509850995100510151025103510451055106510751085109511051115112511351145115511651175118511951205121512251235124512551265127512851295130513151325133513451355136513751385139514051415142514351445145514651475148514951505151515251535154515551565157515851595160516151625163516451655166516751685169517051715172517351745175517651775178517951805181518251835184518551865187518851895190519151925193519451955196519751985199520052015202520352045205520652075208520952105211521252135214521552165217521852195220522152225223522452255226522752285229523052315232523352345235523652375238523952405241524252435244524552465247524852495250525152525253525452555256525752585259526052615262526352645265526652675268526952705271527252735274527552765277527852795280528152825283528452855286528752885289529052915292529352945295529652975298529953005301530253035304530553065307530853095310531153125313531453155316531753185319532053215322532353245325532653275328532953305331533253335334533553365337533853395340534153425343534453455346534753485349535053515352535353545355535653575358535953605361536253635364536553665367536853695370537153725373537453755376537753785379538053815382538353845385538653875388538953905391539253935394539553965397539853995400540154025403540454055406540754085409541054115412541354145415541654175418541954205421542254235424542554265427542854295430543154325433543454355436543754385439544054415442544354445445544654475448544954505451545254535454545554565457545854595460546154625463546454655466546754685469547054715472547354745475547654775478547954805481548254835484548554865487548854895490549154925493549454955496549754985499550055015502550355045505550655075508550955105511551255135514551555165517551855195520552155225523552455255526552755285529553055315532553355345535553655375538553955405541554255435544554555465547554855495550555155525553555455555556555755585559556055615562556355645565556655675568556955705571557255735574557555765577557855795580558155825583558455855586558755885589559055915592559355945595559655975598559956005601560256035604560556065607560856095610561156125613561456155616561756185619562056215622562356245625562656275628562956305631563256335634563556365637563856395640564156425643564456455646564756485649565056515652565356545655565656575658565956605661566256635664566556665667566856695670567156725673567456755676567756785679568056815682568356845685568656875688568956905691569256935694569556965697569856995700570157025703570457055706570757085709571057115712571357145715571657175718571957205721572257235724572557265727572857295730573157325733573457355736573757385739574057415742574357445745574657475748574957505751575257535754575557565757575857595760576157625763576457655766576757685769577057715772577357745775577657775778577957805781578257835784578557865787578857895790579157925793579457955796579757985799580058015802580358045805580658075808580958105811581258135814581558165817581858195820582158225823582458255826582758285829583058315832583358345835583658375838583958405841584258435844584558465847584858495850585158525853585458555856585758585859586058615862586358645865586658675868586958705871587258735874587558765877587858795880588158825883588458855886588758885889589058915892589358945895589658975898589959005901590259035904590559065907590859095910591159125913591459155916591759185919592059215922592359245925592659275928592959305931593259335934593559365937593859395940594159425943594459455946594759485949595059515952595359545955595659575958595959605961596259635964596559665967596859695970597159725973597459755976597759785979598059815982598359845985598659875988598959905991599259935994599559965997599859996000600160026003600460056006600760086009601060116012601360146015601660176018601960206021602260236024602560266027602860296030603160326033603460356036603760386039604060416042604360446045604660476048604960506051605260536054605560566057605860596060606160626063606460656066606760686069607060716072607360746075607660776078607960806081608260836084608560866087608860896090609160926093609460956096609760986099610061016102610361046105610661076108610961106111611261136114611561166117611861196120612161226123612461256126612761286129613061316132613361346135613661376138613961406141614261436144614561466147614861496150615161526153615461556156615761586159616061616162616361646165616661676168616961706171617261736174617561766177617861796180618161826183618461856186618761886189619061916192619361946195619661976198619962006201620262036204620562066207620862096210621162126213621462156216621762186219622062216222622362246225622662276228622962306231623262336234623562366237623862396240624162426243624462456246624762486249625062516252625362546255625662576258625962606261626262636264626562666267626862696270627162726273627462756276627762786279628062816282628362846285628662876288628962906291629262936294629562966297629862996300630163026303630463056306630763086309631063116312631363146315631663176318631963206321632263236324632563266327632863296330633163326333633463356336633763386339634063416342634363446345634663476348634963506351635263536354635563566357635863596360636163626363636463656366636763686369637063716372637363746375637663776378637963806381638263836384638563866387638863896390639163926393639463956396639763986399640064016402640364046405640664076408640964106411641264136414641564166417641864196420642164226423642464256426642764286429643064316432643364346435643664376438643964406441644264436444644564466447644864496450645164526453645464556456645764586459646064616462646364646465646664676468646964706471647264736474647564766477647864796480648164826483648464856486648764886489649064916492649364946495649664976498649965006501650265036504650565066507650865096510651165126513651465156516651765186519652065216522652365246525652665276528652965306531653265336534653565366537653865396540654165426543654465456546654765486549655065516552655365546555655665576558655965606561656265636564656565666567656865696570657165726573657465756576657765786579658065816582658365846585658665876588658965906591659265936594659565966597659865996600660166026603660466056606660766086609661066116612661366146615661666176618661966206621662266236624662566266627662866296630663166326633663466356636663766386639664066416642664366446645664666476648664966506651665266536654665566566657665866596660666166626663666466656666666766686669667066716672667366746675667666776678667966806681668266836684668566866687668866896690669166926693669466956696669766986699670067016702670367046705670667076708670967106711671267136714671567166717671867196720672167226723672467256726672767286729673067316732673367346735673667376738673967406741674267436744674567466747674867496750675167526753675467556756675767586759676067616762676367646765676667676768676967706771677267736774677567766777677867796780678167826783678467856786678767886789679067916792679367946795679667976798679968006801680268036804680568066807680868096810681168126813681468156816681768186819682068216822682368246825682668276828682968306831683268336834683568366837683868396840684168426843684468456846684768486849685068516852685368546855685668576858685968606861686268636864686568666867686868696870687168726873687468756876687768786879688068816882688368846885688668876888688968906891689268936894689568966897689868996900690169026903690469056906690769086909691069116912691369146915691669176918691969206921692269236924692569266927692869296930693169326933693469356936693769386939694069416942694369446945694669476948694969506951695269536954695569566957695869596960696169626963696469656966696769686969697069716972697369746975697669776978697969806981698269836984698569866987698869896990699169926993699469956996699769986999700070017002700370047005700670077008700970107011701270137014701570167017701870197020702170227023702470257026702770287029703070317032703370347035703670377038703970407041704270437044704570467047704870497050705170527053705470557056705770587059706070617062706370647065706670677068706970707071707270737074707570767077707870797080708170827083708470857086708770887089709070917092709370947095709670977098709971007101710271037104710571067107710871097110711171127113711471157116711771187119712071217122712371247125712671277128712971307131713271337134713571367137713871397140714171427143714471457146714771487149715071517152715371547155715671577158715971607161716271637164716571667167716871697170717171727173717471757176717771787179718071817182718371847185718671877188718971907191719271937194719571967197719871997200720172027203720472057206720772087209721072117212721372147215721672177218721972207221722272237224722572267227722872297230723172327233723472357236723772387239724072417242724372447245724672477248724972507251725272537254725572567257725872597260726172627263726472657266726772687269727072717272727372747275727672777278727972807281728272837284728572867287728872897290729172927293729472957296729772987299730073017302730373047305730673077308730973107311731273137314731573167317731873197320732173227323732473257326732773287329733073317332733373347335733673377338733973407341734273437344734573467347734873497350735173527353735473557356735773587359736073617362736373647365736673677368736973707371737273737374737573767377737873797380738173827383738473857386738773887389739073917392739373947395739673977398739974007401740274037404740574067407740874097410741174127413741474157416741774187419742074217422742374247425742674277428742974307431743274337434743574367437743874397440744174427443744474457446744774487449745074517452745374547455745674577458745974607461746274637464746574667467746874697470747174727473747474757476747774787479748074817482748374847485748674877488748974907491749274937494749574967497749874997500750175027503750475057506750775087509751075117512751375147515751675177518751975207521752275237524752575267527752875297530753175327533753475357536753775387539754075417542754375447545754675477548754975507551755275537554755575567557755875597560756175627563756475657566756775687569757075717572757375747575757675777578757975807581758275837584758575867587758875897590759175927593759475957596759775987599760076017602760376047605760676077608760976107611761276137614761576167617761876197620762176227623762476257626762776287629763076317632763376347635763676377638763976407641764276437644764576467647764876497650765176527653765476557656765776587659766076617662766376647665766676677668766976707671767276737674767576767677767876797680768176827683768476857686768776887689769076917692769376947695769676977698769977007701770277037704770577067707770877097710771177127713771477157716771777187719772077217722772377247725772677277728772977307731773277337734773577367737773877397740774177427743774477457746774777487749775077517752775377547755775677577758775977607761776277637764776577667767776877697770777177727773777477757776777777787779778077817782778377847785778677877788778977907791779277937794779577967797779877997800780178027803780478057806780778087809781078117812781378147815781678177818781978207821782278237824782578267827782878297830783178327833783478357836783778387839784078417842784378447845784678477848784978507851785278537854785578567857785878597860786178627863786478657866786778687869787078717872787378747875787678777878787978807881788278837884788578867887788878897890789178927893789478957896789778987899790079017902790379047905790679077908790979107911791279137914791579167917791879197920792179227923792479257926792779287929793079317932793379347935793679377938793979407941794279437944794579467947794879497950795179527953795479557956795779587959796079617962796379647965796679677968796979707971797279737974797579767977797879797980798179827983798479857986798779887989799079917992799379947995799679977998799980008001800280038004800580068007800880098010801180128013801480158016801780188019802080218022802380248025802680278028802980308031803280338034803580368037803880398040804180428043804480458046804780488049805080518052805380548055805680578058805980608061806280638064806580668067806880698070807180728073807480758076807780788079808080818082808380848085808680878088808980908091809280938094809580968097809880998100810181028103810481058106810781088109811081118112811381148115811681178118811981208121812281238124812581268127812881298130813181328133813481358136813781388139814081418142814381448145814681478148814981508151815281538154815581568157815881598160816181628163816481658166816781688169817081718172817381748175817681778178817981808181818281838184818581868187818881898190819181928193819481958196819781988199820082018202820382048205820682078208820982108211821282138214821582168217821882198220822182228223822482258226822782288229823082318232823382348235823682378238823982408241824282438244824582468247824882498250825182528253825482558256825782588259826082618262826382648265826682678268826982708271827282738274827582768277827882798280828182828283828482858286828782888289829082918292829382948295829682978298829983008301830283038304830583068307830883098310831183128313831483158316831783188319832083218322832383248325832683278328832983308331833283338334833583368337833883398340834183428343834483458346834783488349835083518352835383548355835683578358835983608361836283638364836583668367836883698370837183728373837483758376837783788379838083818382838383848385838683878388838983908391839283938394839583968397839883998400840184028403840484058406840784088409841084118412841384148415841684178418841984208421842284238424842584268427842884298430843184328433843484358436843784388439844084418442844384448445844684478448844984508451845284538454845584568457845884598460846184628463846484658466846784688469847084718472847384748475847684778478847984808481848284838484848584868487848884898490849184928493849484958496849784988499850085018502850385048505850685078508850985108511851285138514851585168517851885198520852185228523852485258526852785288529853085318532853385348535853685378538853985408541854285438544854585468547854885498550855185528553855485558556855785588559856085618562856385648565856685678568856985708571857285738574857585768577857885798580858185828583858485858586858785888589859085918592859385948595859685978598859986008601860286038604860586068607860886098610861186128613861486158616861786188619862086218622862386248625862686278628862986308631863286338634863586368637863886398640864186428643864486458646864786488649865086518652865386548655865686578658865986608661866286638664866586668667866886698670867186728673867486758676867786788679868086818682868386848685868686878688868986908691869286938694869586968697869886998700870187028703870487058706870787088709871087118712871387148715871687178718871987208721872287238724872587268727872887298730873187328733873487358736873787388739874087418742874387448745874687478748874987508751875287538754875587568757875887598760876187628763876487658766876787688769877087718772877387748775877687778778877987808781878287838784878587868787878887898790879187928793879487958796879787988799880088018802880388048805880688078808880988108811881288138814881588168817881888198820882188228823882488258826882788288829883088318832883388348835883688378838883988408841884288438844884588468847884888498850885188528853885488558856885788588859886088618862886388648865886688678868886988708871887288738874887588768877887888798880888188828883888488858886888788888889889088918892889388948895889688978898889989008901890289038904890589068907890889098910891189128913891489158916891789188919892089218922892389248925892689278928892989308931893289338934893589368937893889398940894189428943894489458946894789488949895089518952895389548955895689578958895989608961896289638964896589668967896889698970897189728973897489758976897789788979898089818982898389848985898689878988898989908991899289938994899589968997899889999000900190029003900490059006900790089009901090119012901390149015901690179018901990209021902290239024902590269027902890299030903190329033903490359036903790389039904090419042904390449045904690479048904990509051905290539054905590569057905890599060906190629063906490659066906790689069907090719072907390749075907690779078907990809081908290839084908590869087908890899090909190929093909490959096909790989099910091019102910391049105910691079108910991109111911291139114911591169117911891199120912191229123912491259126912791289129913091319132913391349135913691379138913991409141914291439144914591469147914891499150915191529153915491559156915791589159916091619162916391649165916691679168916991709171917291739174917591769177917891799180918191829183918491859186918791889189919091919192919391949195919691979198919992009201920292039204920592069207920892099210921192129213921492159216921792189219922092219222922392249225922692279228922992309231923292339234923592369237923892399240924192429243924492459246924792489249925092519252925392549255925692579258925992609261926292639264926592669267926892699270927192729273927492759276927792789279928092819282928392849285928692879288928992909291929292939294929592969297929892999300930193029303930493059306930793089309931093119312931393149315931693179318931993209321932293239324932593269327932893299330933193329333933493359336933793389339934093419342934393449345934693479348934993509351935293539354935593569357935893599360936193629363936493659366936793689369937093719372937393749375937693779378937993809381938293839384938593869387938893899390939193929393939493959396939793989399940094019402940394049405940694079408940994109411941294139414941594169417941894199420942194229423942494259426942794289429943094319432943394349435943694379438943994409441944294439444944594469447944894499450945194529453945494559456945794589459946094619462946394649465946694679468946994709471947294739474947594769477947894799480948194829483948494859486948794889489949094919492949394949495949694979498949995009501950295039504950595069507950895099510951195129513951495159516951795189519952095219522952395249525952695279528952995309531953295339534953595369537953895399540954195429543954495459546954795489549955095519552955395549555955695579558955995609561956295639564956595669567956895699570957195729573957495759576957795789579958095819582958395849585958695879588958995909591959295939594959595969597959895999600960196029603960496059606960796089609961096119612961396149615961696179618961996209621962296239624962596269627962896299630963196329633963496359636963796389639964096419642964396449645964696479648964996509651965296539654965596569657965896599660966196629663966496659666966796689669967096719672967396749675967696779678967996809681968296839684968596869687968896899690969196929693969496959696969796989699970097019702970397049705970697079708970997109711971297139714971597169717971897199720972197229723972497259726972797289729973097319732973397349735973697379738973997409741974297439744974597469747974897499750975197529753975497559756975797589759976097619762976397649765976697679768976997709771977297739774977597769777977897799780978197829783978497859786978797889789979097919792979397949795979697979798979998009801980298039804980598069807980898099810981198129813981498159816981798189819982098219822982398249825982698279828982998309831983298339834983598369837983898399840984198429843984498459846984798489849985098519852985398549855985698579858985998609861986298639864986598669867986898699870987198729873987498759876987798789879988098819882988398849885988698879888988998909891989298939894989598969897989898999900990199029903990499059906990799089909991099119912991399149915991699179918991999209921992299239924992599269927992899299930993199329933993499359936993799389939994099419942994399449945994699479948994999509951995299539954995599569957995899599960996199629963996499659966996799689969997099719972997399749975997699779978997999809981998299839984998599869987998899899990999199929993999499959996999799989999100001000110002100031000410005100061000710008100091001010011100121001310014100151001610017100181001910020100211002210023100241002510026100271002810029100301003110032100331003410035100361003710038100391004010041100421004310044100451004610047100481004910050100511005210053100541005510056100571005810059100601006110062100631006410065100661006710068100691007010071100721007310074100751007610077100781007910080100811008210083100841008510086100871008810089100901009110092100931009410095100961009710098100991010010101101021010310104101051010610107101081010910110101111011210113101141011510116101171011810119101201012110122101231012410125101261012710128101291013010131101321013310134101351013610137101381013910140101411014210143101441014510146101471014810149101501015110152101531015410155101561015710158101591016010161101621016310164101651016610167101681016910170101711017210173101741017510176101771017810179101801018110182101831018410185101861018710188101891019010191101921019310194101951019610197101981019910200102011020210203 | /***    P R O C E S S I N G . J S - 1.4.1    a port of the Processing visualization language    Processing.js is licensed under the MIT License, see LICENSE.    For a list of copyright holders, please refer to AUTHORS.    http://processingjs.org***/(function(window, document, Math, undef) {  var nop = function() {};  var debug = function() {    if ("console" in window) return function(msg) {      window.console.log("Processing.js: " + msg)    };    return nop  }();  var ajax = function(url) {    var xhr = new XMLHttpRequest;    xhr.open("GET", url, false);    if (xhr.overrideMimeType) xhr.overrideMimeType("text/plain");    xhr.setRequestHeader("If-Modified-Since", "Fri, 01 Jan 1960 00:00:00 GMT");    xhr.send(null);    if (xhr.status !== 200 && xhr.status !== 0) throw "XMLHttpRequest failed, status code " + xhr.status;    return xhr.responseText  };  var isDOMPresent = "document" in this && !("fake" in this.document);  document.head = document.head || document.getElementsByTagName("head")[0];  function setupTypedArray(name, fallback) {    if (name in window) return window[name];    if (typeof window[fallback] === "function") return window[fallback];    return function(obj) {      if (obj instanceof Array) return obj;      if (typeof obj === "number") {        var arr = [];        arr.length = obj;        return arr      }    }  }  if (document.documentMode >= 9 && !document.doctype) throw "The doctype directive is missing. The recommended doctype in Internet Explorer is the HTML5 doctype: <!DOCTYPE html>";  var Float32Array = setupTypedArray("Float32Array", "WebGLFloatArray"),    Int32Array = setupTypedArray("Int32Array", "WebGLIntArray"),    Uint16Array = setupTypedArray("Uint16Array", "WebGLUnsignedShortArray"),    Uint8Array = setupTypedArray("Uint8Array", "WebGLUnsignedByteArray");  var PConstants = {    X: 0,    Y: 1,    Z: 2,    R: 3,    G: 4,    B: 5,    A: 6,    U: 7,    V: 8,    NX: 9,    NY: 10,    NZ: 11,    EDGE: 12,    SR: 13,    SG: 14,    SB: 15,    SA: 16,    SW: 17,    TX: 18,    TY: 19,    TZ: 20,    VX: 21,    VY: 22,    VZ: 23,    VW: 24,    AR: 25,    AG: 26,    AB: 27,    DR: 3,    DG: 4,    DB: 5,    DA: 6,    SPR: 28,    SPG: 29,    SPB: 30,    SHINE: 31,    ER: 32,    EG: 33,    EB: 34,    BEEN_LIT: 35,    VERTEX_FIELD_COUNT: 36,    P2D: 1,    JAVA2D: 1,    WEBGL: 2,    P3D: 2,    OPENGL: 2,    PDF: 0,    DXF: 0,    OTHER: 0,    WINDOWS: 1,    MAXOSX: 2,    LINUX: 3,    EPSILON: 1.0E-4,    MAX_FLOAT: 3.4028235E38,    MIN_FLOAT: -3.4028235E38,    MAX_INT: 2147483647,    MIN_INT: -2147483648,    PI: Math.PI,    TWO_PI: 2 * Math.PI,    HALF_PI: Math.PI / 2,    THIRD_PI: Math.PI / 3,    QUARTER_PI: Math.PI / 4,    DEG_TO_RAD: Math.PI / 180,    RAD_TO_DEG: 180 / Math.PI,    WHITESPACE: " \t\n\r\u000c\u00a0",    RGB: 1,    ARGB: 2,    HSB: 3,    ALPHA: 4,    CMYK: 5,    TIFF: 0,    TARGA: 1,    JPEG: 2,    GIF: 3,    BLUR: 11,    GRAY: 12,    INVERT: 13,    OPAQUE: 14,    POSTERIZE: 15,    THRESHOLD: 16,    ERODE: 17,    DILATE: 18,    REPLACE: 0,    BLEND: 1 << 0,    ADD: 1 << 1,    SUBTRACT: 1 << 2,    LIGHTEST: 1 << 3,    DARKEST: 1 << 4,    DIFFERENCE: 1 << 5,    EXCLUSION: 1 << 6,    MULTIPLY: 1 << 7,    SCREEN: 1 << 8,    OVERLAY: 1 << 9,    HARD_LIGHT: 1 << 10,    SOFT_LIGHT: 1 << 11,    DODGE: 1 << 12,    BURN: 1 << 13,    ALPHA_MASK: 4278190080,    RED_MASK: 16711680,    GREEN_MASK: 65280,    BLUE_MASK: 255,    CUSTOM: 0,    ORTHOGRAPHIC: 2,    PERSPECTIVE: 3,    POINT: 2,    POINTS: 2,    LINE: 4,    LINES: 4,    TRIANGLE: 8,    TRIANGLES: 9,    TRIANGLE_STRIP: 10,    TRIANGLE_FAN: 11,    QUAD: 16,    QUADS: 16,    QUAD_STRIP: 17,    POLYGON: 20,    PATH: 21,    RECT: 30,    ELLIPSE: 31,    ARC: 32,    SPHERE: 40,    BOX: 41,    GROUP: 0,    PRIMITIVE: 1,    GEOMETRY: 3,    VERTEX: 0,    BEZIER_VERTEX: 1,    CURVE_VERTEX: 2,    BREAK: 3,    CLOSESHAPE: 4,    OPEN: 1,    CLOSE: 2,    CORNER: 0,    CORNERS: 1,    RADIUS: 2,    CENTER_RADIUS: 2,    CENTER: 3,    DIAMETER: 3,    CENTER_DIAMETER: 3,    BASELINE: 0,    TOP: 101,    BOTTOM: 102,    NORMAL: 1,    NORMALIZED: 1,    IMAGE: 2,    MODEL: 4,    SHAPE: 5,    SQUARE: "butt",    ROUND: "round",    PROJECT: "square",    MITER: "miter",    BEVEL: "bevel",    AMBIENT: 0,    DIRECTIONAL: 1,    SPOT: 3,    BACKSPACE: 8,    TAB: 9,    ENTER: 10,    RETURN: 13,    ESC: 27,    DELETE: 127,    CODED: 65535,    SHIFT: 16,    CONTROL: 17,    ALT: 18,    CAPSLK: 20,    PGUP: 33,    PGDN: 34,    END: 35,    HOME: 36,    LEFT: 37,    UP: 38,    RIGHT: 39,    DOWN: 40,    F1: 112,    F2: 113,    F3: 114,    F4: 115,    F5: 116,    F6: 117,    F7: 118,    F8: 119,    F9: 120,    F10: 121,    F11: 122,    F12: 123,    NUMLK: 144,    META: 157,    INSERT: 155,    ARROW: "default",    CROSS: "crosshair",    HAND: "pointer",    MOVE: "move",    TEXT: "text",    WAIT: "wait",    NOCURSOR: "url(''), auto",    DISABLE_OPENGL_2X_SMOOTH: 1,    ENABLE_OPENGL_2X_SMOOTH: -1,    ENABLE_OPENGL_4X_SMOOTH: 2,    ENABLE_NATIVE_FONTS: 3,    DISABLE_DEPTH_TEST: 4,    ENABLE_DEPTH_TEST: -4,    ENABLE_DEPTH_SORT: 5,    DISABLE_DEPTH_SORT: -5,    DISABLE_OPENGL_ERROR_REPORT: 6,    ENABLE_OPENGL_ERROR_REPORT: -6,    ENABLE_ACCURATE_TEXTURES: 7,    DISABLE_ACCURATE_TEXTURES: -7,    HINT_COUNT: 10,    SINCOS_LENGTH: 720,    PRECISIONB: 15,    PRECISIONF: 1 << 15,    PREC_MAXVAL: (1 << 15) - 1,    PREC_ALPHA_SHIFT: 24 - 15,    PREC_RED_SHIFT: 16 - 15,    NORMAL_MODE_AUTO: 0,    NORMAL_MODE_SHAPE: 1,    NORMAL_MODE_VERTEX: 2,    MAX_LIGHTS: 8  };  function virtHashCode(obj) {    if (typeof obj === "string") {      var hash = 0;      for (var i = 0; i < obj.length; ++i) hash = hash * 31 + obj.charCodeAt(i) & 4294967295;      return hash    }    if (typeof obj !== "object") return obj & 4294967295;    if (obj.hashCode instanceof Function) return obj.hashCode();    if (obj.$id === undef) obj.$id = Math.floor(Math.random() * 65536) - 32768 << 16 | Math.floor(Math.random() * 65536);    return obj.$id  }  function virtEquals(obj, other) {    if (obj === null || other === null) return obj === null && other === null;    if (typeof obj === "string") return obj === other;    if (typeof obj !== "object") return obj === other;    if (obj.equals instanceof Function) return obj.equals(other);    return obj === other  }  var ObjectIterator = function(obj) {    if (obj.iterator instanceof    Function) return obj.iterator();    if (obj instanceof Array) {      var index = -1;      this.hasNext = function() {        return ++index < obj.length      };      this.next = function() {        return obj[index]      }    } else throw "Unable to iterate: " + obj;  };  var ArrayList = function() {    function Iterator(array) {      var index = 0;      this.hasNext = function() {        return index < array.length      };      this.next = function() {        return array[index++]      };      this.remove = function() {        array.splice(index, 1)      }    }    function ArrayList(a) {      var array;      if (a instanceof ArrayList) array = a.toArray();      else {        array = [];        if (typeof a === "number") array.length = a > 0 ? a : 0      }      this.get = function(i) {        return array[i]      };      this.contains = function(item) {        return this.indexOf(item) > -1      };      this.indexOf = function(item) {        for (var i = 0, len = array.length; i < len; ++i) if (virtEquals(item, array[i])) return i;        return -1      };      this.lastIndexOf = function(item) {        for (var i = array.length - 1; i >= 0; --i) if (virtEquals(item, array[i])) return i;        return -1      };      this.add = function() {        if (arguments.length === 1) array.push(arguments[0]);        else if (arguments.length === 2) {          var arg0 = arguments[0];          if (typeof arg0 === "number") if (arg0 >= 0 && arg0 <= array.length) array.splice(arg0, 0, arguments[1]);          else throw arg0 + " is not a valid index";          else throw typeof arg0 + " is not a number";        } else throw "Please use the proper number of parameters.";      };      this.addAll = function(arg1, arg2) {        var it;        if (typeof arg1 === "number") {          if (arg1 < 0 || arg1 > array.length) throw "Index out of bounds for addAll: " + arg1 + " greater or equal than " + array.length;          it = new ObjectIterator(arg2);          while (it.hasNext()) array.splice(arg1++, 0, it.next())        } else {          it = new ObjectIterator(arg1);          while (it.hasNext()) array.push(it.next())        }      };      this.set = function() {        if (arguments.length === 2) {          var arg0 = arguments[0];          if (typeof arg0 === "number") if (arg0 >= 0 && arg0 < array.length) array.splice(arg0, 1, arguments[1]);          else throw arg0 + " is not a valid index.";          else throw typeof arg0 + " is not a number";        } else throw "Please use the proper number of parameters.";      };      this.size = function() {        return array.length      };      this.clear = function() {        array.length = 0      };      this.remove = function(item) {        if (typeof item === "number") return array.splice(item, 1)[0];        item = this.indexOf(item);        if (item > -1) {          array.splice(item, 1);          return true        }        return false      };      this.removeAll = function(c) {        var i, x, item, newList = new ArrayList;        newList.addAll(this);        this.clear();        for (i = 0, x = 0; i < newList.size(); i++) {          item = newList.get(i);          if (!c.contains(item)) this.add(x++, item)        }        if (this.size() < newList.size()) return true;        return false      };      this.isEmpty = function() {        return !array.length      };      this.clone = function() {        return new ArrayList(this)      };      this.toArray = function() {        return array.slice(0)      };      this.iterator = function() {        return new Iterator(array)      }    }    return ArrayList  }();  var HashMap = function() {    function HashMap() {      if (arguments.length === 1 && arguments[0] instanceof HashMap) return arguments[0].clone();      var initialCapacity = arguments.length > 0 ? arguments[0] : 16;      var loadFactor = arguments.length > 1 ? arguments[1] : 0.75;      var buckets = [];      buckets.length = initialCapacity;      var count = 0;      var hashMap = this;      function getBucketIndex(key) {        var index = virtHashCode(key) % buckets.length;        return index < 0 ? buckets.length + index : index      }      function ensureLoad() {        if (count <= loadFactor * buckets.length) return;        var allEntries = [];        for (var i = 0; i < buckets.length; ++i) if (buckets[i] !== undef) allEntries = allEntries.concat(buckets[i]);        var newBucketsLength = buckets.length * 2;        buckets = [];        buckets.length = newBucketsLength;        for (var j = 0; j < allEntries.length; ++j) {          var index = getBucketIndex(allEntries[j].key);          var bucket = buckets[index];          if (bucket === undef) buckets[index] = bucket = [];          bucket.push(allEntries[j])        }      }      function Iterator(conversion, removeItem) {        var bucketIndex = 0;        var itemIndex = -1;        var endOfBuckets = false;        var currentItem;        function findNext() {          while (!endOfBuckets) {            ++itemIndex;            if (bucketIndex >= buckets.length) endOfBuckets = true;            else if (buckets[bucketIndex] === undef || itemIndex >= buckets[bucketIndex].length) {              itemIndex = -1;              ++bucketIndex            } else return          }        }        this.hasNext = function() {          return !endOfBuckets        };        this.next = function() {          currentItem = conversion(buckets[bucketIndex][itemIndex]);          findNext();          return currentItem        };        this.remove = function() {          if (currentItem !== undef) {            removeItem(currentItem);            --itemIndex;            findNext()          }        };        findNext()      }      function Set(conversion, isIn, removeItem) {        this.clear = function() {          hashMap.clear()        };        this.contains = function(o) {          return isIn(o)        };        this.containsAll = function(o) {          var it = o.iterator();          while (it.hasNext()) if (!this.contains(it.next())) return false;          return true        };        this.isEmpty = function() {          return hashMap.isEmpty()        };        this.iterator = function() {          return new Iterator(conversion, removeItem)        };        this.remove = function(o) {          if (this.contains(o)) {            removeItem(o);            return true          }          return false        };        this.removeAll = function(c) {          var it = c.iterator();          var changed = false;          while (it.hasNext()) {            var item = it.next();            if (this.contains(item)) {              removeItem(item);              changed = true            }          }          return true        };        this.retainAll = function(c) {          var it = this.iterator();          var toRemove = [];          while (it.hasNext()) {            var entry = it.next();            if (!c.contains(entry)) toRemove.push(entry)          }          for (var i = 0; i < toRemove.length; ++i) removeItem(toRemove[i]);          return toRemove.length > 0        };        this.size = function() {          return hashMap.size()        };        this.toArray = function() {          var result = [];          var it = this.iterator();          while (it.hasNext()) result.push(it.next());          return result        }      }      function Entry(pair) {        this._isIn = function(map) {          return map === hashMap && pair.removed === undef        };        this.equals = function(o) {          return virtEquals(pair.key, o.getKey())        };        this.getKey = function() {          return pair.key        };        this.getValue = function() {          return pair.value        };        this.hashCode = function(o) {          return virtHashCode(pair.key)        };        this.setValue = function(value) {          var old = pair.value;          pair.value = value;          return old        }      }      this.clear = function() {        count = 0;        buckets = [];        buckets.length = initialCapacity      };      this.clone = function() {        var map = new HashMap;        map.putAll(this);        return map      };      this.containsKey = function(key) {        var index = getBucketIndex(key);        var bucket = buckets[index];        if (bucket === undef) return false;        for (var i = 0; i < bucket.length; ++i) if (virtEquals(bucket[i].key, key)) return true;        return false      };      this.containsValue = function(value) {        for (var i = 0; i < buckets.length; ++i) {          var bucket = buckets[i];          if (bucket === undef) continue;          for (var j = 0; j < bucket.length; ++j) if (virtEquals(bucket[j].value, value)) return true        }        return false      };      this.entrySet = function() {        return new Set(function(pair) {          return new Entry(pair)        },        function(pair) {          return pair instanceof Entry && pair._isIn(hashMap)        },        function(pair) {          return hashMap.remove(pair.getKey())        })      };      this.get = function(key) {        var index = getBucketIndex(key);        var bucket = buckets[index];        if (bucket === undef) return null;        for (var i = 0; i < bucket.length; ++i) if (virtEquals(bucket[i].key, key)) return bucket[i].value;        return null      };      this.isEmpty = function() {        return count === 0      };      this.keySet = function() {        return new Set(function(pair) {          return pair.key        },        function(key) {          return hashMap.containsKey(key)        },        function(key) {          return hashMap.remove(key)        })      };      this.values = function() {        return new Set(function(pair) {          return pair.value        },        function(value) {          return hashMap.containsValue(value)        },        function(value) {          return hashMap.removeByValue(value)        })      };      this.put = function(key, value) {        var index = getBucketIndex(key);        var bucket = buckets[index];        if (bucket === undef) {          ++count;          buckets[index] = [{            key: key,            value: value          }];          ensureLoad();          return null        }        for (var i = 0; i < bucket.length; ++i) if (virtEquals(bucket[i].key, key)) {          var previous = bucket[i].value;          bucket[i].value = value;          return previous        }++count;        bucket.push({          key: key,          value: value        });        ensureLoad();        return null      };      this.putAll = function(m) {        var it = m.entrySet().iterator();        while (it.hasNext()) {          var entry = it.next();          this.put(entry.getKey(), entry.getValue())        }      };      this.remove = function(key) {        var index = getBucketIndex(key);        var bucket = buckets[index];        if (bucket === undef) return null;        for (var i = 0; i < bucket.length; ++i) if (virtEquals(bucket[i].key, key)) {          --count;          var previous = bucket[i].value;          bucket[i].removed = true;          if (bucket.length > 1) bucket.splice(i, 1);          else buckets[index] = undef;          return previous        }        return null      };      this.removeByValue = function(value) {        var bucket, i, ilen, pair;        for (bucket in buckets) if (buckets.hasOwnProperty(bucket)) for (i = 0, ilen = buckets[bucket].length; i < ilen; i++) {          pair = buckets[bucket][i];          if (pair.value === value) {            buckets[bucket].splice(i, 1);            return true          }        }        return false      };      this.size = function() {        return count      }    }    return HashMap  }();  var PVector = function() {    function PVector(x, y, z) {      this.x = x || 0;      this.y = y || 0;      this.z = z || 0    }    PVector.dist = function(v1, v2) {      return v1.dist(v2)    };    PVector.dot = function(v1, v2) {      return v1.dot(v2)    };    PVector.cross = function(v1, v2) {      return v1.cross(v2)    };    PVector.angleBetween = function(v1, v2) {      return Math.acos(v1.dot(v2) / (v1.mag() * v2.mag()))    };    PVector.prototype = {      set: function(v, y, z) {        if (arguments.length === 1) this.set(v.x || v[0] || 0, v.y || v[1] || 0, v.z || v[2] || 0);        else {          this.x = v;          this.y = y;          this.z = z        }      },      get: function() {        return new PVector(this.x, this.y, this.z)      },      mag: function() {        var x = this.x,          y = this.y,          z = this.z;        return Math.sqrt(x * x + y * y + z * z)      },      add: function(v, y, z) {        if (arguments.length === 1) {          this.x += v.x;          this.y += v.y;          this.z += v.z        } else {          this.x += v;          this.y += y;          this.z += z        }      },      sub: function(v, y, z) {        if (arguments.length === 1) {          this.x -= v.x;          this.y -= v.y;          this.z -= v.z        } else {          this.x -= v;          this.y -= y;          this.z -= z        }      },      mult: function(v) {        if (typeof v === "number") {          this.x *= v;          this.y *= v;          this.z *= v        } else {          this.x *= v.x;          this.y *= v.y;          this.z *= v.z        }      },      div: function(v) {        if (typeof v === "number") {          this.x /= v;          this.y /= v;          this.z /= v        } else {          this.x /= v.x;          this.y /= v.y;          this.z /= v.z        }      },      dist: function(v) {        var dx = this.x - v.x,          dy = this.y - v.y,          dz = this.z - v.z;        return Math.sqrt(dx * dx + dy * dy + dz * dz)      },      dot: function(v, y, z) {        if (arguments.length === 1) return this.x * v.x + this.y * v.y + this.z * v.z;        return this.x * v + this.y * y + this.z * z      },      cross: function(v) {        var x = this.x,          y = this.y,          z = this.z;        return new PVector(y * v.z - v.y * z, z * v.x - v.z * x, x * v.y - v.x * y)      },      normalize: function() {        var m = this.mag();        if (m > 0) this.div(m)      },      limit: function(high) {        if (this.mag() > high) {          this.normalize();          this.mult(high)        }      },      heading2D: function() {        return -Math.atan2(-this.y, this.x)      },      toString: function() {        return "[" + this.x + ", " + this.y + ", " + this.z + "]"      },      array: function() {        return [this.x, this.y, this.z]      }    };    function createPVectorMethod(method) {      return function(v1, v2) {        var v = v1.get();        v[method](v2);        return v      }    }    for (var method in PVector.prototype) if (PVector.prototype.hasOwnProperty(method) && !PVector.hasOwnProperty(method)) PVector[method] = createPVectorMethod(method);    return PVector  }();  function DefaultScope() {}  DefaultScope.prototype = PConstants;  var defaultScope = new DefaultScope;  defaultScope.ArrayList = ArrayList;  defaultScope.HashMap = HashMap;  defaultScope.PVector = PVector;  defaultScope.ObjectIterator = ObjectIterator;  defaultScope.PConstants = PConstants;  defaultScope.defineProperty = function(obj, name, desc) {    if ("defineProperty" in Object) Object.defineProperty(obj, name, desc);    else {      if (desc.hasOwnProperty("get")) obj.__defineGetter__(name, desc.get);      if (desc.hasOwnProperty("set")) obj.__defineSetter__(name, desc.set)    }  };  function overloadBaseClassFunction(object, name, basefn) {    if (!object.hasOwnProperty(name) || typeof object[name] !== "function") {      object[name] = basefn;      return    }    var fn = object[name];    if ("$overloads" in fn) {      fn.$defaultOverload = basefn;      return    }    if (! ("$overloads" in basefn) && fn.length === basefn.length) return;    var overloads, defaultOverload;    if ("$overloads" in basefn) {      overloads = basefn.$overloads.slice(0);      overloads[fn.length] = fn;      defaultOverload = basefn.$defaultOverload    } else {      overloads = [];      overloads[basefn.length] = basefn;      overloads[fn.length] = fn;      defaultOverload = fn    }    var hubfn = function() {      var fn = hubfn.$overloads[arguments.length] || ("$methodArgsIndex" in hubfn && arguments.length > hubfn.$methodArgsIndex ? hubfn.$overloads[hubfn.$methodArgsIndex] : null) || hubfn.$defaultOverload;      return fn.apply(this, arguments)    };    hubfn.$overloads = overloads;    if ("$methodArgsIndex" in basefn) hubfn.$methodArgsIndex = basefn.$methodArgsIndex;    hubfn.$defaultOverload = defaultOverload;    hubfn.name = name;    object[name] = hubfn  }  function extendClass(subClass, baseClass) {    function extendGetterSetter(propertyName) {      defaultScope.defineProperty(subClass, propertyName, {        get: function() {          return baseClass[propertyName]        },        set: function(v) {          baseClass[propertyName] = v        },        enumerable: true      })    }    var properties = [];    for (var propertyName in baseClass) if (typeof baseClass[propertyName] === "function") overloadBaseClassFunction(subClass, propertyName, baseClass[propertyName]);    else if (propertyName.charAt(0) !== "$" && !(propertyName in subClass)) properties.push(propertyName);    while (properties.length > 0) extendGetterSetter(properties.shift());    subClass.$super = baseClass  }  defaultScope.extendClassChain = function(base) {    var path = [base];    for (var self = base.$upcast; self; self = self.$upcast) {      extendClass(self, base);      path.push(self);      base = self    }    while (path.length > 0) path.pop().$self = base  };  defaultScope.extendStaticMembers = function(derived, base) {    extendClass(derived, base)  };  defaultScope.extendInterfaceMembers = function(derived, base) {    extendClass(derived, base)  };  defaultScope.addMethod = function(object, name, fn, hasMethodArgs) {    var existingfn = object[name];    if (existingfn || hasMethodArgs) {      var args = fn.length;      if ("$overloads" in existingfn) existingfn.$overloads[args] = fn;      else {        var hubfn = function() {          var fn = hubfn.$overloads[arguments.length] || ("$methodArgsIndex" in hubfn && arguments.length > hubfn.$methodArgsIndex ? hubfn.$overloads[hubfn.$methodArgsIndex] : null) || hubfn.$defaultOverload;          return fn.apply(this, arguments)        };        var overloads = [];        if (existingfn) overloads[existingfn.length] = existingfn;        overloads[args] = fn;        hubfn.$overloads = overloads;        hubfn.$defaultOverload = existingfn || fn;        if (hasMethodArgs) hubfn.$methodArgsIndex = args;        hubfn.name = name;        object[name] = hubfn      }    } else object[name] = fn  };  function isNumericalJavaType(type) {    if (typeof type !== "string") return false;    return ["byte", "int", "char", "color", "float", "long", "double"].indexOf(type) !== -1  }  defaultScope.createJavaArray = function(type, bounds) {    var result = null,      defaultValue = null;    if (typeof type === "string") if (type === "boolean") defaultValue = false;    else if (isNumericalJavaType(type)) defaultValue = 0;    if (typeof bounds[0] === "number") {      var itemsCount = 0 | bounds[0];      if (bounds.length <= 1) {        result = [];        result.length = itemsCount;        for (var i = 0; i < itemsCount; ++i) result[i] = defaultValue      } else {        result = [];        var newBounds = bounds.slice(1);        for (var j = 0; j < itemsCount; ++j) result.push(defaultScope.createJavaArray(type, newBounds))      }    }    return result  };  var colors = {    aliceblue: "#f0f8ff",    antiquewhite: "#faebd7",    aqua: "#00ffff",    aquamarine: "#7fffd4",    azure: "#f0ffff",    beige: "#f5f5dc",    bisque: "#ffe4c4",    black: "#000000",    blanchedalmond: "#ffebcd",    blue: "#0000ff",    blueviolet: "#8a2be2",    brown: "#a52a2a",    burlywood: "#deb887",    cadetblue: "#5f9ea0",    chartreuse: "#7fff00",    chocolate: "#d2691e",    coral: "#ff7f50",    cornflowerblue: "#6495ed",    cornsilk: "#fff8dc",    crimson: "#dc143c",    cyan: "#00ffff",    darkblue: "#00008b",    darkcyan: "#008b8b",    darkgoldenrod: "#b8860b",    darkgray: "#a9a9a9",    darkgreen: "#006400",    darkkhaki: "#bdb76b",    darkmagenta: "#8b008b",    darkolivegreen: "#556b2f",    darkorange: "#ff8c00",    darkorchid: "#9932cc",    darkred: "#8b0000",    darksalmon: "#e9967a",    darkseagreen: "#8fbc8f",    darkslateblue: "#483d8b",    darkslategray: "#2f4f4f",    darkturquoise: "#00ced1",    darkviolet: "#9400d3",    deeppink: "#ff1493",    deepskyblue: "#00bfff",    dimgray: "#696969",    dodgerblue: "#1e90ff",    firebrick: "#b22222",    floralwhite: "#fffaf0",    forestgreen: "#228b22",    fuchsia: "#ff00ff",    gainsboro: "#dcdcdc",    ghostwhite: "#f8f8ff",    gold: "#ffd700",    goldenrod: "#daa520",    gray: "#808080",    green: "#008000",    greenyellow: "#adff2f",    honeydew: "#f0fff0",    hotpink: "#ff69b4",    indianred: "#cd5c5c",    indigo: "#4b0082",    ivory: "#fffff0",    khaki: "#f0e68c",    lavender: "#e6e6fa",    lavenderblush: "#fff0f5",    lawngreen: "#7cfc00",    lemonchiffon: "#fffacd",    lightblue: "#add8e6",    lightcoral: "#f08080",    lightcyan: "#e0ffff",    lightgoldenrodyellow: "#fafad2",    lightgrey: "#d3d3d3",    lightgreen: "#90ee90",    lightpink: "#ffb6c1",    lightsalmon: "#ffa07a",    lightseagreen: "#20b2aa",    lightskyblue: "#87cefa",    lightslategray: "#778899",    lightsteelblue: "#b0c4de",    lightyellow: "#ffffe0",    lime: "#00ff00",    limegreen: "#32cd32",    linen: "#faf0e6",    magenta: "#ff00ff",    maroon: "#800000",    mediumaquamarine: "#66cdaa",    mediumblue: "#0000cd",    mediumorchid: "#ba55d3",    mediumpurple: "#9370d8",    mediumseagreen: "#3cb371",    mediumslateblue: "#7b68ee",    mediumspringgreen: "#00fa9a",    mediumturquoise: "#48d1cc",    mediumvioletred: "#c71585",    midnightblue: "#191970",    mintcream: "#f5fffa",    mistyrose: "#ffe4e1",    moccasin: "#ffe4b5",    navajowhite: "#ffdead",    navy: "#000080",    oldlace: "#fdf5e6",    olive: "#808000",    olivedrab: "#6b8e23",    orange: "#ffa500",    orangered: "#ff4500",    orchid: "#da70d6",    palegoldenrod: "#eee8aa",    palegreen: "#98fb98",    paleturquoise: "#afeeee",    palevioletred: "#d87093",    papayawhip: "#ffefd5",    peachpuff: "#ffdab9",    peru: "#cd853f",    pink: "#ffc0cb",    plum: "#dda0dd",    powderblue: "#b0e0e6",    purple: "#800080",    red: "#ff0000",    rosybrown: "#bc8f8f",    royalblue: "#4169e1",    saddlebrown: "#8b4513",    salmon: "#fa8072",    sandybrown: "#f4a460",    seagreen: "#2e8b57",    seashell: "#fff5ee",    sienna: "#a0522d",    silver: "#c0c0c0",    skyblue: "#87ceeb",    slateblue: "#6a5acd",    slategray: "#708090",    snow: "#fffafa",    springgreen: "#00ff7f",    steelblue: "#4682b4",    tan: "#d2b48c",    teal: "#008080",    thistle: "#d8bfd8",    tomato: "#ff6347",    turquoise: "#40e0d0",    violet: "#ee82ee",    wheat: "#f5deb3",    white: "#ffffff",    whitesmoke: "#f5f5f5",    yellow: "#ffff00",    yellowgreen: "#9acd32"  };  (function(Processing) {    var unsupportedP5 = ("open() createOutput() createInput() BufferedReader selectFolder() " + "dataPath() createWriter() selectOutput() beginRecord() " + "saveStream() endRecord() selectInput() saveBytes() createReader() " + "beginRaw() endRaw() PrintWriter delay()").split(" "),      count = unsupportedP5.length,      prettyName, p5Name;    function createUnsupportedFunc(n) {      return function() {        throw "Processing.js does not support " + n + ".";      }    }    while (count--) {      prettyName = unsupportedP5[count];      p5Name = prettyName.replace("()", "");      Processing[p5Name] = createUnsupportedFunc(prettyName)    }  })(defaultScope);  defaultScope.defineProperty(defaultScope, "screenWidth", {    get: function() {      return window.innerWidth    }  });  defaultScope.defineProperty(defaultScope, "screenHeight", {    get: function() {      return window.innerHeight    }  });  defaultScope.defineProperty(defaultScope, "online", {    get: function() {      return true    }  });  var processingInstances = [];  var processingInstanceIds = {};  var removeInstance = function(id) {    processingInstances.splice(processingInstanceIds[id], 1);    delete processingInstanceIds[id]  };  var addInstance = function(processing) {    if (processing.externals.canvas.id === undef || !processing.externals.canvas.id.length) processing.externals.canvas.id = "__processing" + processingInstances.length;    processingInstanceIds[processing.externals.canvas.id] = processingInstances.length;    processingInstances.push(processing)  };  function computeFontMetrics(pfont) {    var emQuad = 250,      correctionFactor = pfont.size / emQuad,      canvas = document.createElement("canvas");    canvas.width = 2 * emQuad;    canvas.height = 2 * emQuad;    canvas.style.opacity = 0;    var cfmFont = pfont.getCSSDefinition(emQuad + "px", "normal"),      ctx = canvas.getContext("2d");    ctx.font = cfmFont;    var protrusions = "dbflkhyjqpg";    canvas.width = ctx.measureText(protrusions).width;    ctx.font = cfmFont;    var leadDiv = document.createElement("div");    leadDiv.style.position = "absolute";    leadDiv.style.opacity = 0;    leadDiv.style.fontFamily = '"' + pfont.name + '"';    leadDiv.style.fontSize = emQuad + "px";    leadDiv.innerHTML = protrusions + "<br/>" + protrusions;    document.body.appendChild(leadDiv);    var w = canvas.width,      h = canvas.height,      baseline = h / 2;    ctx.fillStyle = "white";    ctx.fillRect(0, 0, w, h);    ctx.fillStyle = "black";    ctx.fillText(protrusions, 0, baseline);    var pixelData = ctx.getImageData(0, 0, w, h).data;    var i = 0,      w4 = w * 4,      len = pixelData.length;    while (++i < len && pixelData[i] === 255) nop();    var ascent = Math.round(i / w4);    i = len - 1;    while (--i > 0 && pixelData[i] === 255) nop();    var descent = Math.round(i / w4);    pfont.ascent = correctionFactor * (baseline - ascent);    pfont.descent = correctionFactor * (descent - baseline);    if (document.defaultView.getComputedStyle) {      var leadDivHeight = document.defaultView.getComputedStyle(leadDiv, null).getPropertyValue("height");      leadDivHeight = correctionFactor * leadDivHeight.replace("px", "");      if (leadDivHeight >= pfont.size * 2) pfont.leading = Math.round(leadDivHeight / 2)    }    document.body.removeChild(leadDiv);    if (pfont.caching) return ctx  }  function PFont(name, size) {    if (name === undef) name = "";    this.name = name;    if (size === undef) size = 0;    this.size = size;    this.glyph = false;    this.ascent = 0;    this.descent = 0;    this.leading = 1.2 * size;    var illegalIndicator = name.indexOf(" Italic Bold");    if (illegalIndicator !== -1) name = name.substring(0, illegalIndicator);    this.style = "normal";    var italicsIndicator = name.indexOf(" Italic");    if (italicsIndicator !== -1) {      name = name.substring(0, italicsIndicator);      this.style = "italic"    }    this.weight = "normal";    var boldIndicator = name.indexOf(" Bold");    if (boldIndicator !== -1) {      name = name.substring(0, boldIndicator);      this.weight = "bold"    }    this.family = "sans-serif";    if (name !== undef) switch (name) {    case "sans-serif":    case "serif":    case "monospace":    case "fantasy":    case "cursive":      this.family = name;      break;    default:      this.family = '"' + name + '", sans-serif';      break    }    this.context2d = computeFontMetrics(this);    this.css = this.getCSSDefinition();    if (this.context2d) this.context2d.font = this.css  }  PFont.prototype.caching = true;  PFont.prototype.getCSSDefinition = function(fontSize, lineHeight) {    if (fontSize === undef) fontSize = this.size + "px";    if (lineHeight === undef) lineHeight = this.leading + "px";    var components = [this.style, "normal", this.weight, fontSize + "/" + lineHeight, this.family];    return components.join(" ")  };  PFont.prototype.measureTextWidth = function(string) {    return this.context2d.measureText(string).width  };  PFont.prototype.measureTextWidthFallback = function(string) {    var canvas = document.createElement("canvas"),      ctx = canvas.getContext("2d");    ctx.font = this.css;    return ctx.measureText(string).width  };  PFont.PFontCache = {    length: 0  };  PFont.get = function(fontName, fontSize) {    fontSize = (fontSize * 10 + 0.5 | 0) / 10;    var cache = PFont.PFontCache,      idx = fontName + "/" + fontSize;    if (!cache[idx]) {      cache[idx] = new PFont(fontName, fontSize);      cache.length++;      if (cache.length === 50) {        PFont.prototype.measureTextWidth = PFont.prototype.measureTextWidthFallback;        PFont.prototype.caching = false;        var entry;        for (entry in cache) if (entry !== "length") cache[entry].context2d = null;        return new PFont(fontName, fontSize)      }      if (cache.length === 400) {        PFont.PFontCache = {};        PFont.get = PFont.getFallback;        return new PFont(fontName, fontSize)      }    }    return cache[idx]  };  PFont.getFallback = function(fontName, fontSize) {    return new PFont(fontName, fontSize)  };  PFont.list = function() {    return ["sans-serif", "serif", "monospace", "fantasy", "cursive"]  };  PFont.preloading = {    template: {},    initialized: false,    initialize: function() {      var generateTinyFont = function() {        var encoded = "#E3KAI2wAgT1MvMg7Eo3VmNtYX7ABi3CxnbHlm" + "7Abw3kaGVhZ7ACs3OGhoZWE7A53CRobXR47AY3" + "AGbG9jYQ7G03Bm1heH7ABC3CBuYW1l7Ae3AgcG" + "9zd7AI3AE#B3AQ2kgTY18PPPUACwAg3ALSRoo3" + "#yld0xg32QAB77#E777773B#E3C#I#Q77773E#" + "Q7777777772CMAIw7AB77732B#M#Q3wAB#g3B#" + "E#E2BB//82BB////w#B7#gAEg3E77x2B32B#E#" + "Q#MTcBAQ32gAe#M#QQJ#E32M#QQJ#I#g32Q77#";        var expand = function(input) {          return "AAAAAAAA".substr(~~input ? 7 - input : 6)        };        return encoded.replace(/[#237]/g, expand)      };      var fontface = document.createElement("style");      fontface.setAttribute("type", "text/css");      fontface.innerHTML = "@font-face {\n" + '  font-family: "PjsEmptyFont";' + "\n" + "  src: url('data:application/x-font-ttf;base64," + generateTinyFont() + "')\n" + "       format('truetype');\n" + "}";      document.head.appendChild(fontface);      var element = document.createElement("span");      element.style.cssText = 'position: absolute; top: 0; left: 0; opacity: 0; font-family: "PjsEmptyFont", fantasy;';      element.innerHTML = "AAAAAAAA";      document.body.appendChild(element);      this.template = element;      this.initialized = true    },    getElementWidth: function(element) {      return document.defaultView.getComputedStyle(element, "").getPropertyValue("width")    },    timeAttempted: 0,    pending: function(intervallength) {      if (!this.initialized) this.initialize();      var element, computedWidthFont, computedWidthRef = this.getElementWidth(this.template);      for (var i = 0; i < this.fontList.length; i++) {        element = this.fontList[i];        computedWidthFont = this.getElementWidth(element);        if (this.timeAttempted < 4E3 && computedWidthFont === computedWidthRef) {          this.timeAttempted += intervallength;          return true        } else {          document.body.removeChild(element);          this.fontList.splice(i--, 1);          this.timeAttempted = 0        }      }      if (this.fontList.length === 0) return false;      return true    },    fontList: [],    addedList: {},    add: function(fontSrc) {      if (!this.initialized) this.initialize();      var fontName = typeof fontSrc === "object" ? fontSrc.fontFace : fontSrc,      fontUrl = typeof fontSrc === "object" ? fontSrc.url : fontSrc;      if (this.addedList[fontName]) return;      var style = document.createElement("style");      style.setAttribute("type", "text/css");      style.innerHTML = "@font-face{\n  font-family: '" + fontName + "';\n  src:  url('" + fontUrl + "');\n}\n";      document.head.appendChild(style);      this.addedList[fontName] = true;      var element = document.createElement("span");      element.style.cssText = "position: absolute; top: 0; left: 0; opacity: 0;";      element.style.fontFamily = '"' + fontName + '", "PjsEmptyFont", fantasy';      element.innerHTML = "AAAAAAAA";      document.body.appendChild(element);      this.fontList.push(element)    }  };  defaultScope.PFont = PFont;  var Processing = this.Processing = function(aCanvas, aCode) {    if (! (this instanceof    Processing)) throw "called Processing constructor as if it were a function: missing 'new'.";    var curElement, pgraphicsMode = aCanvas === undef && aCode === undef;    if (pgraphicsMode) curElement = document.createElement("canvas");    else curElement = typeof aCanvas === "string" ? document.getElementById(aCanvas) : aCanvas;    if (! (curElement instanceof HTMLCanvasElement)) throw "called Processing constructor without passing canvas element reference or id.";    function unimplemented(s) {      Processing.debug("Unimplemented - " + s)    }    var p = this;    p.externals = {      canvas: curElement,      context: undef,      sketch: undef    };    p.name = "Processing.js Instance";    p.use3DContext = false;    p.focused = false;    p.breakShape = false;    p.glyphTable = {};    p.pmouseX = 0;    p.pmouseY = 0;    p.mouseX = 0;    p.mouseY = 0;    p.mouseButton = 0;    p.mouseScroll = 0;    p.mouseClicked = undef;    p.mouseDragged = undef;    p.mouseMoved = undef;    p.mousePressed = undef;    p.mouseReleased = undef;    p.mouseScrolled = undef;    p.mouseOver = undef;    p.mouseOut = undef;    p.touchStart = undef;    p.touchEnd = undef;    p.touchMove = undef;    p.touchCancel = undef;    p.key = undef;    p.keyCode = undef;    p.keyPressed = nop;    p.keyReleased = nop;    p.keyTyped = nop;    p.draw = undef;    p.setup = undef;    p.__mousePressed = false;    p.__keyPressed = false;    p.__frameRate = 60;    p.frameCount = 0;    p.width = 100;    p.height = 100;    var curContext, curSketch, drawing, online = true,      doFill = true,      fillStyle = [1, 1, 1, 1],      currentFillColor = 4294967295,      isFillDirty = true,      doStroke = true,      strokeStyle = [0, 0, 0, 1],      currentStrokeColor = 4278190080,      isStrokeDirty = true,      lineWidth = 1,      loopStarted = false,      renderSmooth = false,      doLoop = true,      looping = 0,      curRectMode = 0,      curEllipseMode = 3,      normalX = 0,      normalY = 0,      normalZ = 0,      normalMode = 0,      curFrameRate = 60,      curMsPerFrame = 1E3 / curFrameRate,      curCursor = 'default',      oldCursor = curElement.style.cursor,      curShape = 20,      curShapeCount = 0,      curvePoints = [],      curTightness = 0,      curveDet = 20,      curveInited = false,      backgroundObj = -3355444,      bezDetail = 20,      colorModeA = 255,      colorModeX = 255,      colorModeY = 255,      colorModeZ = 255,      pathOpen = false,      mouseDragging = false,      pmouseXLastFrame = 0,      pmouseYLastFrame = 0,      curColorMode = 1,      curTint = null,      curTint3d = null,      getLoaded = false,      start = Date.now(),      timeSinceLastFPS = start,      framesSinceLastFPS = 0,      textcanvas, curveBasisMatrix, curveToBezierMatrix, curveDrawMatrix, bezierDrawMatrix, bezierBasisInverse, bezierBasisMatrix, curContextCache = {      attributes: {},      locations: {}    },      programObject3D, programObject2D, programObjectUnlitShape, boxBuffer, boxNormBuffer, boxOutlineBuffer, rectBuffer, rectNormBuffer, sphereBuffer, lineBuffer, fillBuffer, fillColorBuffer, strokeColorBuffer, pointBuffer, shapeTexVBO, canTex, textTex, curTexture = {      width: 0,      height: 0    },      curTextureMode = 2,      usingTexture = false,      textBuffer, textureBuffer, indexBuffer, horizontalTextAlignment = 37,      verticalTextAlignment = 0,      textMode = 4,      curFontName = "Arial",      curTextSize = 12,      curTextAscent = 9,      curTextDescent = 2,      curTextLeading = 14,      curTextFont = PFont.get(curFontName, curTextSize),      originalContext, proxyContext = null,      isContextReplaced = false,      setPixelsCached, maxPixelsCached = 1E3,      pressedKeysMap = [],      lastPressedKeyCode = null,      codedKeys = [16,      17, 18, 20, 33, 34, 35, 36, 37, 38, 39, 40, 144, 155, 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 157];    var stylePaddingLeft, stylePaddingTop, styleBorderLeft, styleBorderTop;    if (document.defaultView && document.defaultView.getComputedStyle) {      stylePaddingLeft = parseInt(document.defaultView.getComputedStyle(curElement, null)["paddingLeft"], 10) || 0;      stylePaddingTop = parseInt(document.defaultView.getComputedStyle(curElement, null)["paddingTop"], 10) || 0;      styleBorderLeft = parseInt(document.defaultView.getComputedStyle(curElement, null)["borderLeftWidth"], 10) || 0;      styleBorderTop = parseInt(document.defaultView.getComputedStyle(curElement, null)["borderTopWidth"], 10) || 0    }    var lightCount = 0;    var sphereDetailV = 0,      sphereDetailU = 0,      sphereX = [],      sphereY = [],      sphereZ = [],      sinLUT = new Float32Array(720),      cosLUT = new Float32Array(720),      sphereVerts, sphereNorms;    var cam, cameraInv, modelView, modelViewInv, userMatrixStack, userReverseMatrixStack, inverseCopy, projection, manipulatingCamera = false,      frustumMode = false,      cameraFOV = 60 * (Math.PI / 180),      cameraX = p.width / 2,      cameraY = p.height / 2,      cameraZ = cameraY / Math.tan(cameraFOV / 2),      cameraNear = cameraZ / 10,      cameraFar = cameraZ * 10,      cameraAspect = p.width / p.height;    var vertArray = [],      curveVertArray = [],      curveVertCount = 0,      isCurve = false,      isBezier = false,      firstVert = true;    var curShapeMode = 0;    var styleArray = [];    var boxVerts = new Float32Array([0.5, 0.5, -0.5, 0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, 0.5, -0.5, 0.5, 0.5, -0.5, 0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, 0.5, -0.5, 0.5, 0.5, 0.5, 0.5, 0.5, 0.5, -0.5, 0.5, 0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, -0.5, 0.5, 0.5, -0.5, 0.5, -0.5, -0.5, 0.5, -0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, -0.5, 0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, 0.5, -0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, 0.5, -0.5, -0.5, -0.5, -0.5, 0.5, 0.5, 0.5, 0.5, 0.5, -0.5, -0.5,      0.5, -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, 0.5, 0.5, 0.5, 0.5]);    var boxOutlineVerts = new Float32Array([0.5, 0.5, 0.5, 0.5, -0.5, 0.5, 0.5, 0.5, -0.5, 0.5, -0.5, -0.5, -0.5, 0.5, -0.5, -0.5, -0.5, -0.5, -0.5, 0.5, 0.5, -0.5, -0.5, 0.5, 0.5, 0.5, 0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, 0.5, -0.5, 0.5, 0.5, 0.5, 0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, -0.5, 0.5, -0.5, -0.5, 0.5, 0.5, -0.5, 0.5]);    var boxNorms = new Float32Array([0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, 1, 0, 0, 1, 0,      0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, -1, 0, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1, 0]);    var rectVerts = new Float32Array([0, 0, 0, 0, 1, 0, 1, 1, 0, 1, 0, 0]);    var rectNorms = new Float32Array([0, 0, 1, 0, 0, 1, 0, 0, 1, 0, 0, 1]);    var vertexShaderSrcUnlitShape = "varying vec4 vFrontColor;" + "attribute vec3 aVertex;" + "attribute vec4 aColor;" + "uniform mat4 uView;" + "uniform mat4 uProjection;" + "uniform float uPointSize;" + "void main(void) {" + "  vFrontColor = aColor;" + "  gl_PointSize = uPointSize;" + "  gl_Position = uProjection * uView * vec4(aVertex, 1.0);" + "}";    var fragmentShaderSrcUnlitShape = "#ifdef GL_ES\n" + "precision highp float;\n" + "#endif\n" + "varying vec4 vFrontColor;" + "uniform bool uSmooth;" + "void main(void){" + "  if(uSmooth == true){" + "    float dist = distance(gl_PointCoord, vec2(0.5));" + "    if(dist > 0.5){" + "      discard;" + "    }" + "  }" + "  gl_FragColor = vFrontColor;" + "}";    var vertexShaderSrc2D = "varying vec4 vFrontColor;" + "attribute vec3 aVertex;" + "attribute vec2 aTextureCoord;" + "uniform vec4 uColor;" + "uniform mat4 uModel;" + "uniform mat4 uView;" + "uniform mat4 uProjection;" + "uniform float uPointSize;" + "varying vec2 vTextureCoord;" + "void main(void) {" + "  gl_PointSize = uPointSize;" + "  vFrontColor = uColor;" + "  gl_Position = uProjection * uView * uModel * vec4(aVertex, 1.0);" + "  vTextureCoord = aTextureCoord;" + "}";    var fragmentShaderSrc2D = "#ifdef GL_ES\n" + "precision highp float;\n" + "#endif\n" + "varying vec4 vFrontColor;" + "varying vec2 vTextureCoord;" + "uniform sampler2D uSampler;" + "uniform int uIsDrawingText;" + "uniform bool uSmooth;" + "void main(void){" + "  if(uSmooth == true){" + "    float dist = distance(gl_PointCoord, vec2(0.5));" + "    if(dist > 0.5){" + "      discard;" + "    }" + "  }" + "  if(uIsDrawingText == 1){" + "    float alpha = texture2D(uSampler, vTextureCoord).a;" + "    gl_FragColor = vec4(vFrontColor.rgb * alpha, alpha);" + "  }" + "  else{" + "    gl_FragColor = vFrontColor;" + "  }" + "}";    var webglMaxTempsWorkaround = /Windows/.test(navigator.userAgent);    var vertexShaderSrc3D = "varying vec4 vFrontColor;" + "attribute vec3 aVertex;" + "attribute vec3 aNormal;" + "attribute vec4 aColor;" + "attribute vec2 aTexture;" + "varying   vec2 vTexture;" + "uniform vec4 uColor;" + "uniform bool uUsingMat;" + "uniform vec3 uSpecular;" + "uniform vec3 uMaterialEmissive;" + "uniform vec3 uMaterialAmbient;" + "uniform vec3 uMaterialSpecular;" + "uniform float uShininess;" + "uniform mat4 uModel;" + "uniform mat4 uView;" + "uniform mat4 uProjection;" + "uniform mat4 uNormalTransform;" + "uniform int uLightCount;" + "uniform vec3 uFalloff;" + "struct Light {" + "  int type;" + "  vec3 color;" + "  vec3 position;" + "  vec3 direction;" + "  float angle;" + "  vec3 halfVector;" + "  float concentration;" + "};" + "uniform Light uLights0;" + "uniform Light uLights1;" + "uniform Light uLights2;" + "uniform Light uLights3;" + "uniform Light uLights4;" + "uniform Light uLights5;" + "uniform Light uLights6;" + "uniform Light uLights7;" + "Light getLight(int index){" + "  if(index == 0) return uLights0;" + "  if(index == 1) return uLights1;" + "  if(index == 2) return uLights2;" + "  if(index == 3) return uLights3;" + "  if(index == 4) return uLights4;" + "  if(index == 5) return uLights5;" + "  if(index == 6) return uLights6;" + "  return uLights7;" + "}" + "void AmbientLight( inout vec3 totalAmbient, in vec3 ecPos, in Light light ) {" + "  float d = length( light.position - ecPos );" + "  float attenuation = 1.0 / ( uFalloff[0] + ( uFalloff[1] * d ) + ( uFalloff[2] * d * d ));" + "  totalAmbient += light.color * attenuation;" + "}" + "void DirectionalLight( inout vec3 col, inout vec3 spec, in vec3 vertNormal, in vec3 ecPos, in Light light ) {" + "  float powerFactor = 0.0;" + "  float nDotVP = max(0.0, dot( vertNormal, normalize(-light.position) ));" + "  float nDotVH = max(0.0, dot( vertNormal, normalize(-light.position-normalize(ecPos) )));" + "  if( nDotVP != 0.0 ){" + "    powerFactor = pow( nDotVH, uShininess );" + "  }" + "  col += light.color * nDotVP;" + "  spec += uSpecular * powerFactor;" + "}" + "void PointLight( inout vec3 col, inout vec3 spec, in vec3 vertNormal, in vec3 ecPos, in Light light ) {" + "  float powerFactor;" + "   vec3 VP = light.position - ecPos;" + "  float d = length( VP ); " + "  VP = normalize( VP );" + "  float attenuation = 1.0 / ( uFalloff[0] + ( uFalloff[1] * d ) + ( uFalloff[2] * d * d ));" + "  float nDotVP = max( 0.0, dot( vertNormal, VP ));" + "  vec3 halfVector = normalize( VP - normalize(ecPos) );" + "  float nDotHV = max( 0.0, dot( vertNormal, halfVector ));" + "  if( nDotVP == 0.0 ) {" + "    powerFactor = 0.0;" + "  }" + "  else {" + "    powerFactor = pow( nDotHV, uShininess );" + "  }" + "  spec += uSpecular * powerFactor * attenuation;" + "  col += light.color * nDotVP * attenuation;" + "}" + "void SpotLight( inout vec3 col, inout vec3 spec, in vec3 vertNormal, in vec3 ecPos, in Light light ) {" + "  float spotAttenuation;" + "  float powerFactor = 0.0;" + "  vec3 VP = light.position - ecPos;" + "  vec3 ldir = normalize( -light.direction );" + "  float d = length( VP );" + "  VP = normalize( VP );" + "  float attenuation = 1.0 / ( uFalloff[0] + ( uFalloff[1] * d ) + ( uFalloff[2] * d * d ) );" + "  float spotDot = dot( VP, ldir );" + (webglMaxTempsWorkaround ? "  spotAttenuation = 1.0; " : "  if( spotDot > cos( light.angle ) ) {" + "    spotAttenuation = pow( spotDot, light.concentration );" + "  }" + "  else{" + "    spotAttenuation = 0.0;" + "  }" + "  attenuation *= spotAttenuation;" + "") + "  float nDotVP = max( 0.0, dot( vertNormal, VP ) );" + "  vec3 halfVector = normalize( VP - normalize(ecPos) );" + "  float nDotHV = max( 0.0, dot( vertNormal, halfVector ) );" + "  if( nDotVP != 0.0 ) {" + "    powerFactor = pow( nDotHV, uShininess );" + "  }" + "  spec += uSpecular * powerFactor * attenuation;" + "  col += light.color * nDotVP * attenuation;" + "}" + "void main(void) {" + "  vec3 finalAmbient = vec3( 0.0 );" + "  vec3 finalDiffuse = vec3( 0.0 );" + "  vec3 finalSpecular = vec3( 0.0 );" + "  vec4 col = uColor;" + "  if ( uColor[0] == -1.0 ){" + "    col = aColor;" + "  }" + "  vec3 norm = normalize(vec3( uNormalTransform * vec4( aNormal, 0.0 ) ));" + "  vec4 ecPos4 = uView * uModel * vec4(aVertex, 1.0);" + "  vec3 ecPos = (vec3(ecPos4))/ecPos4.w;" + "  if( uLightCount == 0 ) {" + "    vFrontColor = col + vec4(uMaterialSpecular, 1.0);" + "  }" + "  else {" + "    for( int i = 0; i < 8; i++ ) {" + "      Light l = getLight(i);" + "      if( i >= uLightCount ){" + "        break;" + "      }" + "      if( l.type == 0 ) {" + "        AmbientLight( finalAmbient, ecPos, l );" + "      }" + "      else if( l.type == 1 ) {" + "        DirectionalLight( finalDiffuse, finalSpecular, norm, ecPos, l );" + "      }" + "      else if( l.type == 2 ) {" + "        PointLight( finalDiffuse, finalSpecular, norm, ecPos, l );" + "      }" + "      else {" + "        SpotLight( finalDiffuse, finalSpecular, norm, ecPos, l );" + "      }" + "    }" + "   if( uUsingMat == false ) {" + "     vFrontColor = vec4(" + "       vec3( col ) * finalAmbient +" + "       vec3( col ) * finalDiffuse +" + "       vec3( col ) * finalSpecular," + "       col[3] );" + "   }" + "   else{" + "     vFrontColor = vec4( " + "       uMaterialEmissive + " + "       (vec3(col) * uMaterialAmbient * finalAmbient ) + " + "       (vec3(col) * finalDiffuse) + " + "       (uMaterialSpecular * finalSpecular), " + "       col[3] );" + "    }" + "  }" + "  vTexture.xy = aTexture.xy;" + "  gl_Position = uProjection * uView * uModel * vec4( aVertex, 1.0 );" + "}";    var fragmentShaderSrc3D = "#ifdef GL_ES\n" + "precision highp float;\n" + "#endif\n" + "varying vec4 vFrontColor;" + "uniform sampler2D uSampler;" + "uniform bool uUsingTexture;" + "varying vec2 vTexture;" + "void main(void){" + "  if( uUsingTexture ){" + "    gl_FragColor = vec4(texture2D(uSampler, vTexture.xy)) * vFrontColor;" + "  }" + "  else{" + "    gl_FragColor = vFrontColor;" + "  }" + "}";    function uniformf(cacheId, programObj, varName, varValue) {      var varLocation = curContextCache.locations[cacheId];      if (varLocation === undef) {        varLocation = curContext.getUniformLocation(programObj, varName);        curContextCache.locations[cacheId] = varLocation      }      if (varLocation !== null) if (varValue.length === 4) curContext.uniform4fv(varLocation, varValue);      else if (varValue.length === 3) curContext.uniform3fv(varLocation, varValue);      else if (varValue.length === 2) curContext.uniform2fv(varLocation, varValue);      else curContext.uniform1f(varLocation, varValue)    }    function uniformi(cacheId, programObj, varName, varValue) {      var varLocation = curContextCache.locations[cacheId];      if (varLocation === undef) {        varLocation = curContext.getUniformLocation(programObj, varName);        curContextCache.locations[cacheId] = varLocation      }      if (varLocation !== null) if (varValue.length === 4) curContext.uniform4iv(varLocation, varValue);      else if (varValue.length === 3) curContext.uniform3iv(varLocation, varValue);      else if (varValue.length === 2) curContext.uniform2iv(varLocation, varValue);      else curContext.uniform1i(varLocation, varValue)    }    function uniformMatrix(cacheId, programObj, varName, transpose, matrix) {      var varLocation = curContextCache.locations[cacheId];      if (varLocation === undef) {        varLocation = curContext.getUniformLocation(programObj, varName);        curContextCache.locations[cacheId] = varLocation      }      if (varLocation !== -1) if (matrix.length === 16) curContext.uniformMatrix4fv(varLocation, transpose, matrix);      else if (matrix.length === 9) curContext.uniformMatrix3fv(varLocation, transpose, matrix);      else curContext.uniformMatrix2fv(varLocation, transpose, matrix)    }    function vertexAttribPointer(cacheId, programObj, varName, size, VBO) {      var varLocation = curContextCache.attributes[cacheId];      if (varLocation === undef) {        varLocation = curContext.getAttribLocation(programObj, varName);        curContextCache.attributes[cacheId] = varLocation      }      if (varLocation !== -1) {        curContext.bindBuffer(curContext.ARRAY_BUFFER, VBO);        curContext.vertexAttribPointer(varLocation, size, curContext.FLOAT, false, 0, 0);        curContext.enableVertexAttribArray(varLocation)      }    }    function disableVertexAttribPointer(cacheId, programObj, varName) {      var varLocation = curContextCache.attributes[cacheId];      if (varLocation === undef) {        varLocation = curContext.getAttribLocation(programObj, varName);        curContextCache.attributes[cacheId] = varLocation      }      if (varLocation !== -1) curContext.disableVertexAttribArray(varLocation)    }    var createProgramObject = function(curContext, vetexShaderSource, fragmentShaderSource) {      var vertexShaderObject = curContext.createShader(curContext.VERTEX_SHADER);      curContext.shaderSource(vertexShaderObject, vetexShaderSource);      curContext.compileShader(vertexShaderObject);      if (!curContext.getShaderParameter(vertexShaderObject, curContext.COMPILE_STATUS)) throw curContext.getShaderInfoLog(vertexShaderObject);      var fragmentShaderObject = curContext.createShader(curContext.FRAGMENT_SHADER);      curContext.shaderSource(fragmentShaderObject, fragmentShaderSource);      curContext.compileShader(fragmentShaderObject);      if (!curContext.getShaderParameter(fragmentShaderObject, curContext.COMPILE_STATUS)) throw curContext.getShaderInfoLog(fragmentShaderObject);      var programObject = curContext.createProgram();      curContext.attachShader(programObject, vertexShaderObject);      curContext.attachShader(programObject, fragmentShaderObject);      curContext.linkProgram(programObject);      if (!curContext.getProgramParameter(programObject, curContext.LINK_STATUS)) throw "Error linking shaders.";      return programObject    };    var imageModeCorner = function(x, y, w, h, whAreSizes) {      return {        x: x,        y: y,        w: w,        h: h      }    };    var imageModeConvert = imageModeCorner;    var imageModeCorners = function(x, y, w, h, whAreSizes) {      return {        x: x,        y: y,        w: whAreSizes ? w : w - x,        h: whAreSizes ? h : h - y      }    };    var imageModeCenter = function(x, y, w, h, whAreSizes) {      return {        x: x - w / 2,        y: y - h / 2,        w: w,        h: h      }    };    var DrawingShared = function() {};    var Drawing2D = function() {};    var Drawing3D = function() {};    var DrawingPre = function() {};    Drawing2D.prototype = new DrawingShared;    Drawing2D.prototype.constructor = Drawing2D;    Drawing3D.prototype = new DrawingShared;    Drawing3D.prototype.constructor = Drawing3D;    DrawingPre.prototype = new DrawingShared;    DrawingPre.prototype.constructor = DrawingPre;    DrawingShared.prototype.a3DOnlyFunction = nop;    var charMap = {};    var Char = p.Character = function(chr) {      if (typeof chr === "string" && chr.length === 1) this.code = chr.charCodeAt(0);      else if (typeof chr === "number") this.code = chr;      else if (chr instanceof Char) this.code = chr;      else this.code = NaN;      return charMap[this.code] === undef ? charMap[this.code] = this : charMap[this.code]    };    Char.prototype.toString = function() {      return String.fromCharCode(this.code)    };    Char.prototype.valueOf = function() {      return this.code    };    var PShape = p.PShape = function(family) {      this.family = family || 0;      this.visible = true;      this.style = true;      this.children = [];      this.nameTable = [];      this.params = [];      this.name = "";      this.image = null;      this.matrix = null;      this.kind = null;      this.close = null;      this.width = null;      this.height = null;      this.parent = null    };    PShape.prototype = {      isVisible: function() {        return this.visible      },      setVisible: function(visible) {        this.visible = visible      },      disableStyle: function() {        this.style = false;        for (var i = 0, j = this.children.length; i < j; i++) this.children[i].disableStyle()      },      enableStyle: function() {        this.style = true;        for (var i = 0, j = this.children.length; i < j; i++) this.children[i].enableStyle()      },      getFamily: function() {        return this.family      },      getWidth: function() {        return this.width      },      getHeight: function() {        return this.height      },      setName: function(name) {        this.name = name      },      getName: function() {        return this.name      },      draw: function(renderContext) {        renderContext = renderContext || p;        if (this.visible) {          this.pre(renderContext);          this.drawImpl(renderContext);          this.post(renderContext)        }      },      drawImpl: function(renderContext) {        if (this.family === 0) this.drawGroup(renderContext);        else if (this.family === 1) this.drawPrimitive(renderContext);        else if (this.family === 3) this.drawGeometry(renderContext);        else if (this.family === 21) this.drawPath(renderContext)      },      drawPath: function(renderContext) {        var i, j;        if (this.vertices.length === 0) return;        renderContext.beginShape();        if (this.vertexCodes.length === 0) if (this.vertices[0].length === 2) for (i = 0, j = this.vertices.length; i < j; i++) renderContext.vertex(this.vertices[i][0], this.vertices[i][1]);        else for (i = 0, j = this.vertices.length; i < j; i++) renderContext.vertex(this.vertices[i][0], this.vertices[i][1], this.vertices[i][2]);        else {          var index = 0;          if (this.vertices[0].length === 2) for (i = 0, j = this.vertexCodes.length; i < j; i++) if (this.vertexCodes[i] === 0) {            renderContext.vertex(this.vertices[index][0], this.vertices[index][1], this.vertices[index]["moveTo"]);            renderContext.breakShape = false;            index++          } else if (this.vertexCodes[i] === 1) {            renderContext.bezierVertex(this.vertices[index + 0][0], this.vertices[index + 0][1], this.vertices[index + 1][0], this.vertices[index + 1][1], this.vertices[index + 2][0], this.vertices[index + 2][1]);            index += 3          } else if (this.vertexCodes[i] === 2) {            renderContext.curveVertex(this.vertices[index][0], this.vertices[index][1]);            index++          } else {            if (this.vertexCodes[i] === 3) renderContext.breakShape = true          } else for (i = 0, j = this.vertexCodes.length; i < j; i++) if (this.vertexCodes[i] === 0) {            renderContext.vertex(this.vertices[index][0], this.vertices[index][1], this.vertices[index][2]);            if (this.vertices[index]["moveTo"] === true) vertArray[vertArray.length - 1]["moveTo"] = true;            else if (this.vertices[index]["moveTo"] === false) vertArray[vertArray.length - 1]["moveTo"] = false;            renderContext.breakShape = false          } else if (this.vertexCodes[i] === 1) {            renderContext.bezierVertex(this.vertices[index + 0][0], this.vertices[index + 0][1], this.vertices[index + 0][2], this.vertices[index + 1][0], this.vertices[index + 1][1], this.vertices[index + 1][2], this.vertices[index + 2][0], this.vertices[index + 2][1], this.vertices[index + 2][2]);            index += 3          } else if (this.vertexCodes[i] === 2) {            renderContext.curveVertex(this.vertices[index][0], this.vertices[index][1], this.vertices[index][2]);            index++          } else if (this.vertexCodes[i] === 3) renderContext.breakShape = true        }        renderContext.endShape(this.close ? 2 : 1)      },      drawGeometry: function(renderContext) {        var i, j;        renderContext.beginShape(this.kind);        if (this.style) for (i = 0, j = this.vertices.length; i < j; i++) renderContext.vertex(this.vertices[i]);        else for (i = 0, j = this.vertices.length; i < j; i++) {          var vert = this.vertices[i];          if (vert[2] === 0) renderContext.vertex(vert[0], vert[1]);          else renderContext.vertex(vert[0], vert[1], vert[2])        }        renderContext.endShape()      },      drawGroup: function(renderContext) {        for (var i = 0, j = this.children.length; i < j; i++) this.children[i].draw(renderContext)      },      drawPrimitive: function(renderContext) {        if (this.kind === 2) renderContext.point(this.params[0], this.params[1]);        else if (this.kind === 4) if (this.params.length === 4) renderContext.line(this.params[0], this.params[1], this.params[2], this.params[3]);        else renderContext.line(this.params[0], this.params[1], this.params[2], this.params[3], this.params[4], this.params[5]);        else if (this.kind === 8) renderContext.triangle(this.params[0], this.params[1], this.params[2], this.params[3], this.params[4], this.params[5]);        else if (this.kind === 16) renderContext.quad(this.params[0], this.params[1], this.params[2], this.params[3], this.params[4], this.params[5], this.params[6], this.params[7]);        else if (this.kind === 30) if (this.image !== null) {          var imMode = imageModeConvert;          renderContext.imageMode(0);          renderContext.image(this.image, this.params[0], this.params[1], this.params[2], this.params[3]);          imageModeConvert = imMode        } else {          var rcMode = curRectMode;          renderContext.rectMode(0);          renderContext.rect(this.params[0], this.params[1], this.params[2], this.params[3]);          curRectMode = rcMode        } else if (this.kind === 31) {          var elMode = curEllipseMode;          renderContext.ellipseMode(0);          renderContext.ellipse(this.params[0], this.params[1], this.params[2], this.params[3]);          curEllipseMode = elMode        } else if (this.kind === 32) {          var eMode = curEllipseMode;          renderContext.ellipseMode(0);          renderContext.arc(this.params[0], this.params[1], this.params[2], this.params[3], this.params[4], this.params[5]);          curEllipseMode = eMode        } else if (this.kind === 41) if (this.params.length === 1) renderContext.box(this.params[0]);        else renderContext.box(this.params[0], this.params[1], this.params[2]);        else if (this.kind === 40) renderContext.sphere(this.params[0])      },      pre: function(renderContext) {        if (this.matrix) {          renderContext.pushMatrix();          renderContext.transform(this.matrix)        }        if (this.style) {          renderContext.pushStyle();          this.styles(renderContext)        }      },      post: function(renderContext) {        if (this.matrix) renderContext.popMatrix();        if (this.style) renderContext.popStyle()      },      styles: function(renderContext) {        if (this.stroke) {          renderContext.stroke(this.strokeColor);          renderContext.strokeWeight(this.strokeWeight);          renderContext.strokeCap(this.strokeCap);          renderContext.strokeJoin(this.strokeJoin)        } else renderContext.noStroke();        if (this.fill) renderContext.fill(this.fillColor);        else renderContext.noFill()      },      getChild: function(child) {        var i, j;        if (typeof child === "number") return this.children[child];        var found;        if (child === "" || this.name === child) return this;        if (this.nameTable.length > 0) {          for (i = 0, j = this.nameTable.length; i < j || found; i++) if (this.nameTable[i].getName === child) {            found = this.nameTable[i];            break          }          if (found) return found        }        for (i = 0, j = this.children.length; i < j; i++) {          found = this.children[i].getChild(child);          if (found) return found        }        return null      },      getChildCount: function() {        return this.children.length      },      addChild: function(child) {        this.children.push(child);        child.parent = this;        if (child.getName() !== null) this.addName(child.getName(), child)      },      addName: function(name, shape) {        if (this.parent !== null) this.parent.addName(name, shape);        else this.nameTable.push([name, shape])      },      translate: function() {        if (arguments.length === 2) {          this.checkMatrix(2);          this.matrix.translate(arguments[0], arguments[1])        } else {          this.checkMatrix(3);          this.matrix.translate(arguments[0], arguments[1], 0)        }      },      checkMatrix: function(dimensions) {        if (this.matrix === null) if (dimensions === 2) this.matrix = new p.PMatrix2D;        else this.matrix = new p.PMatrix3D;        else if (dimensions === 3 && this.matrix instanceof p.PMatrix2D) this.matrix = new p.PMatrix3D      },      rotateX: function(angle) {        this.rotate(angle, 1, 0, 0)      },      rotateY: function(angle) {        this.rotate(angle, 0, 1, 0)      },      rotateZ: function(angle) {        this.rotate(angle, 0, 0, 1)      },      rotate: function() {        if (arguments.length === 1) {          this.checkMatrix(2);          this.matrix.rotate(arguments[0])        } else {          this.checkMatrix(3);          this.matrix.rotate(arguments[0], arguments[1], arguments[2], arguments[3])        }      },      scale: function() {        if (arguments.length === 2) {          this.checkMatrix(2);          this.matrix.scale(arguments[0], arguments[1])        } else if (arguments.length === 3) {          this.checkMatrix(2);          this.matrix.scale(arguments[0], arguments[1], arguments[2])        } else {          this.checkMatrix(2);          this.matrix.scale(arguments[0])        }      },      resetMatrix: function() {        this.checkMatrix(2);        this.matrix.reset()      },      applyMatrix: function(matrix) {        if (arguments.length === 1) this.applyMatrix(matrix.elements[0], matrix.elements[1], 0, matrix.elements[2], matrix.elements[3], matrix.elements[4], 0, matrix.elements[5], 0, 0, 1, 0, 0, 0, 0, 1);        else if (arguments.length === 6) {          this.checkMatrix(2);          this.matrix.apply(arguments[0], arguments[1], arguments[2], 0, arguments[3], arguments[4], arguments[5], 0, 0, 0, 1, 0, 0, 0, 0, 1)        } else if (arguments.length === 16) {          this.checkMatrix(3);          this.matrix.apply(arguments[0], arguments[1], arguments[2], arguments[3], arguments[4], arguments[5], arguments[6], arguments[7], arguments[8], arguments[9], arguments[10], arguments[11], arguments[12], arguments[13], arguments[14], arguments[15])        }      }    };    var PShapeSVG = p.PShapeSVG = function() {      p.PShape.call(this);      if (arguments.length === 1) {        this.element = arguments[0];        this.vertexCodes = [];        this.vertices = [];        this.opacity = 1;        this.stroke = false;        this.strokeColor = 4278190080;        this.strokeWeight = 1;        this.strokeCap = 'butt';        this.strokeJoin = 'miter';        this.strokeGradient = null;        this.strokeGradientPaint = null;        this.strokeName = null;        this.strokeOpacity = 1;        this.fill = true;        this.fillColor = 4278190080;        this.fillGradient = null;        this.fillGradientPaint = null;        this.fillName = null;        this.fillOpacity = 1;        if (this.element.getName() !== "svg") throw "root is not <svg>, it's <" + this.element.getName() + ">";      } else if (arguments.length === 2) if (typeof arguments[1] === "string") {        if (arguments[1].indexOf(".svg") > -1) {          this.element = new p.XMLElement(p, arguments[1]);          this.vertexCodes = [];          this.vertices = [];          this.opacity = 1;          this.stroke = false;          this.strokeColor = 4278190080;          this.strokeWeight = 1;          this.strokeCap = 'butt';          this.strokeJoin = 'miter';          this.strokeGradient = "";          this.strokeGradientPaint = "";          this.strokeName = "";          this.strokeOpacity = 1;          this.fill = true;          this.fillColor = 4278190080;          this.fillGradient = null;          this.fillGradientPaint = null;          this.fillOpacity = 1        }      } else if (arguments[0]) {        this.element = arguments[1];        this.vertexCodes = arguments[0].vertexCodes.slice();        this.vertices = arguments[0].vertices.slice();        this.stroke = arguments[0].stroke;        this.strokeColor = arguments[0].strokeColor;        this.strokeWeight = arguments[0].strokeWeight;        this.strokeCap = arguments[0].strokeCap;        this.strokeJoin = arguments[0].strokeJoin;        this.strokeGradient = arguments[0].strokeGradient;        this.strokeGradientPaint = arguments[0].strokeGradientPaint;        this.strokeName = arguments[0].strokeName;        this.fill = arguments[0].fill;        this.fillColor = arguments[0].fillColor;        this.fillGradient = arguments[0].fillGradient;        this.fillGradientPaint = arguments[0].fillGradientPaint;        this.fillName = arguments[0].fillName;        this.strokeOpacity = arguments[0].strokeOpacity;        this.fillOpacity = arguments[0].fillOpacity;        this.opacity = arguments[0].opacity      }      this.name = this.element.getStringAttribute("id");      var displayStr = this.element.getStringAttribute("display", "inline");      this.visible = displayStr !== "none";      var str = this.element.getAttribute("transform");      if (str) this.matrix = this.parseMatrix(str);      var viewBoxStr = this.element.getStringAttribute("viewBox");      if (viewBoxStr !== null) {        var viewBox = viewBoxStr.split(" ");        this.width = viewBox[2];        this.height = viewBox[3]      }      var unitWidth = this.element.getStringAttribute("width");      var unitHeight = this.element.getStringAttribute("height");      if (unitWidth !== null) {        this.width = this.parseUnitSize(unitWidth);        this.height = this.parseUnitSize(unitHeight)      } else if (this.width === 0 || this.height === 0) {        this.width = 1;        this.height = 1;        throw "The width and/or height is not " + "readable in the <svg> tag of this file.";      }      this.parseColors(this.element);      this.parseChildren(this.element)    };    PShapeSVG.prototype = new PShape;    PShapeSVG.prototype.parseMatrix = function() {      function getCoords(s) {        var m = [];        s.replace(/\((.*?)\)/, function() {          return function(all, params) {            m = params.replace(/,+/g, " ").split(/\s+/)          }        }());        return m      }      return function(str) {        this.checkMatrix(2);        var pieces = [];        str.replace(/\s*(\w+)\((.*?)\)/g, function(all) {          pieces.push(p.trim(all))        });        if (pieces.length === 0) return null;        for (var i = 0, j = pieces.length; i < j; i++) {          var m = getCoords(pieces[i]);          if (pieces[i].indexOf("matrix") !== -1) this.matrix.set(m[0], m[2], m[4], m[1], m[3], m[5]);          else if (pieces[i].indexOf("translate") !== -1) {            var tx = m[0];            var ty = m.length === 2 ? m[1] : 0;            this.matrix.translate(tx, ty)          } else if (pieces[i].indexOf("scale") !== -1) {            var sx = m[0];            var sy = m.length === 2 ? m[1] : m[0];            this.matrix.scale(sx, sy)          } else if (pieces[i].indexOf("rotate") !== -1) {            var angle = m[0];            if (m.length === 1) this.matrix.rotate(p.radians(angle));            else if (m.length === 3) {              this.matrix.translate(m[1], m[2]);              this.matrix.rotate(p.radians(m[0]));              this.matrix.translate(-m[1], -m[2])            }          } else if (pieces[i].indexOf("skewX") !== -1) this.matrix.skewX(parseFloat(m[0]));          else if (pieces[i].indexOf("skewY") !== -1) this.matrix.skewY(m[0]);          else if (pieces[i].indexOf("shearX") !== -1) this.matrix.shearX(m[0]);          else if (pieces[i].indexOf("shearY") !== -1) this.matrix.shearY(m[0])        }        return this.matrix      }    }();    PShapeSVG.prototype.parseChildren = function(element) {      var newelement = element.getChildren();      var children = new p.PShape;      for (var i = 0, j = newelement.length; i < j; i++) {        var kid = this.parseChild(newelement[i]);        if (kid) children.addChild(kid)      }      this.children.push(children)    };    PShapeSVG.prototype.getName = function() {      return this.name    };    PShapeSVG.prototype.parseChild = function(elem) {      var name = elem.getName();      var shape;      if (name === "g") shape = new PShapeSVG(this, elem);      else if (name === "defs") shape = new PShapeSVG(this, elem);      else if (name === "line") {        shape = new PShapeSVG(this, elem);        shape.parseLine()      } else if (name === "circle") {        shape = new PShapeSVG(this, elem);        shape.parseEllipse(true)      } else if (name === "ellipse") {        shape = new PShapeSVG(this, elem);        shape.parseEllipse(false)      } else if (name === "rect") {        shape = new PShapeSVG(this, elem);        shape.parseRect()      } else if (name === "polygon") {        shape = new PShapeSVG(this, elem);        shape.parsePoly(true)      } else if (name === "polyline") {        shape = new PShapeSVG(this, elem);        shape.parsePoly(false)      } else if (name === "path") {        shape = new PShapeSVG(this, elem);        shape.parsePath()      } else if (name === "radialGradient") unimplemented("PShapeSVG.prototype.parseChild, name = radialGradient");      else if (name === "linearGradient") unimplemented("PShapeSVG.prototype.parseChild, name = linearGradient");      else if (name === "text") unimplemented("PShapeSVG.prototype.parseChild, name = text");      else if (name === "filter") unimplemented("PShapeSVG.prototype.parseChild, name = filter");      else if (name === "mask") unimplemented("PShapeSVG.prototype.parseChild, name = mask");      else nop();      return shape    };    PShapeSVG.prototype.parsePath = function() {      this.family = 21;      this.kind = 0;      var pathDataChars = [];      var c;      var pathData = p.trim(this.element.getStringAttribute("d").replace(/[\s,]+/g, " "));      if (pathData === null) return;      pathData = p.__toCharArray(pathData);      var cx = 0,        cy = 0,        ctrlX = 0,        ctrlY = 0,        ctrlX1 = 0,        ctrlX2 = 0,        ctrlY1 = 0,        ctrlY2 = 0,        endX = 0,        endY = 0,        ppx = 0,        ppy = 0,        px = 0,        py = 0,        i = 0,        valOf = 0;      var str = "";      var tmpArray = [];      var flag = false;      var lastInstruction;      var command;      var j, k;      while (i < pathData.length) {        valOf = pathData[i].valueOf();        if (valOf >= 65 && valOf <= 90 || valOf >= 97 && valOf <= 122) {          j = i;          i++;          if (i < pathData.length) {            tmpArray = [];            valOf = pathData[i].valueOf();            while (! (valOf >= 65 && valOf <= 90 || valOf >= 97 && valOf <= 100 || valOf >= 102 && valOf <= 122) && flag === false) {              if (valOf === 32) {                if (str !== "") {                  tmpArray.push(parseFloat(str));                  str = ""                }                i++              } else if (valOf === 45) if (pathData[i - 1].valueOf() === 101) {                str += pathData[i].toString();                i++              } else {                if (str !== "") tmpArray.push(parseFloat(str));                str = pathData[i].toString();                i++              } else {                str += pathData[i].toString();                i++              }              if (i === pathData.length) flag = true;              else valOf = pathData[i].valueOf()            }          }          if (str !== "") {            tmpArray.push(parseFloat(str));            str = ""          }          command = pathData[j];          valOf = command.valueOf();          if (valOf === 77) {            if (tmpArray.length >= 2 && tmpArray.length % 2 === 0) {              cx = tmpArray[0];              cy = tmpArray[1];              this.parsePathMoveto(cx, cy);              if (tmpArray.length > 2) for (j = 2, k = tmpArray.length; j < k; j += 2) {                cx = tmpArray[j];                cy = tmpArray[j + 1];                this.parsePathLineto(cx, cy)              }            }          } else if (valOf === 109) {            if (tmpArray.length >= 2 && tmpArray.length % 2 === 0) {              cx += tmpArray[0];              cy += tmpArray[1];              this.parsePathMoveto(cx, cy);              if (tmpArray.length > 2) for (j = 2, k = tmpArray.length; j < k; j += 2) {                cx += tmpArray[j];                cy += tmpArray[j + 1];                this.parsePathLineto(cx, cy)              }            }          } else if (valOf === 76) {            if (tmpArray.length >= 2 && tmpArray.length % 2 === 0) for (j = 0, k = tmpArray.length; j < k; j += 2) {              cx = tmpArray[j];              cy = tmpArray[j + 1];              this.parsePathLineto(cx, cy)            }          } else if (valOf === 108) {            if (tmpArray.length >= 2 && tmpArray.length % 2 === 0) for (j = 0, k = tmpArray.length; j < k; j += 2) {              cx += tmpArray[j];              cy += tmpArray[j + 1];              this.parsePathLineto(cx, cy)            }          } else if (valOf === 72) for (j = 0, k = tmpArray.length; j < k; j++) {            cx = tmpArray[j];            this.parsePathLineto(cx, cy)          } else if (valOf === 104) for (j = 0, k = tmpArray.length; j < k; j++) {            cx += tmpArray[j];            this.parsePathLineto(cx, cy)          } else if (valOf === 86) for (j = 0, k = tmpArray.length; j < k; j++) {            cy = tmpArray[j];            this.parsePathLineto(cx, cy)          } else if (valOf === 118) for (j = 0, k = tmpArray.length; j < k; j++) {            cy += tmpArray[j];            this.parsePathLineto(cx, cy)          } else if (valOf === 67) {            if (tmpArray.length >= 6 && tmpArray.length % 6 === 0) for (j = 0, k = tmpArray.length; j < k; j += 6) {              ctrlX1 = tmpArray[j];              ctrlY1 = tmpArray[j + 1];              ctrlX2 = tmpArray[j + 2];              ctrlY2 = tmpArray[j + 3];              endX = tmpArray[j + 4];              endY = tmpArray[j + 5];              this.parsePathCurveto(ctrlX1, ctrlY1, ctrlX2, ctrlY2, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 99) {            if (tmpArray.length >= 6 && tmpArray.length % 6 === 0) for (j = 0, k = tmpArray.length; j < k; j += 6) {              ctrlX1 = cx + tmpArray[j];              ctrlY1 = cy + tmpArray[j + 1];              ctrlX2 = cx + tmpArray[j + 2];              ctrlY2 = cy + tmpArray[j + 3];              endX = cx + tmpArray[j + 4];              endY = cy + tmpArray[j + 5];              this.parsePathCurveto(ctrlX1, ctrlY1, ctrlX2, ctrlY2, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 83) {            if (tmpArray.length >= 4 && tmpArray.length % 4 === 0) for (j = 0, k = tmpArray.length; j < k; j += 4) {              if (lastInstruction.toLowerCase() === "c" || lastInstruction.toLowerCase() === "s") {                ppx = this.vertices[this.vertices.length - 2][0];                ppy = this.vertices[this.vertices.length - 2][1];                px = this.vertices[this.vertices.length - 1][0];                py = this.vertices[this.vertices.length - 1][1];                ctrlX1 = px + (px - ppx);                ctrlY1 = py + (py - ppy)              } else {                ctrlX1 = this.vertices[this.vertices.length - 1][0];                ctrlY1 = this.vertices[this.vertices.length - 1][1]              }              ctrlX2 = tmpArray[j];              ctrlY2 = tmpArray[j + 1];              endX = tmpArray[j + 2];              endY = tmpArray[j + 3];              this.parsePathCurveto(ctrlX1, ctrlY1, ctrlX2, ctrlY2, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 115) {            if (tmpArray.length >= 4 && tmpArray.length % 4 === 0) for (j = 0, k = tmpArray.length; j < k; j += 4) {              if (lastInstruction.toLowerCase() === "c" || lastInstruction.toLowerCase() === "s") {                ppx = this.vertices[this.vertices.length - 2][0];                ppy = this.vertices[this.vertices.length - 2][1];                px = this.vertices[this.vertices.length - 1][0];                py = this.vertices[this.vertices.length - 1][1];                ctrlX1 = px + (px - ppx);                ctrlY1 = py + (py - ppy)              } else {                ctrlX1 = this.vertices[this.vertices.length - 1][0];                ctrlY1 = this.vertices[this.vertices.length - 1][1]              }              ctrlX2 = cx + tmpArray[j];              ctrlY2 = cy + tmpArray[j + 1];              endX = cx + tmpArray[j + 2];              endY = cy + tmpArray[j + 3];              this.parsePathCurveto(ctrlX1, ctrlY1, ctrlX2, ctrlY2, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 81) {            if (tmpArray.length >= 4 && tmpArray.length % 4 === 0) for (j = 0, k = tmpArray.length; j < k; j += 4) {              ctrlX = tmpArray[j];              ctrlY = tmpArray[j + 1];              endX = tmpArray[j + 2];              endY = tmpArray[j + 3];              this.parsePathQuadto(cx, cy, ctrlX, ctrlY, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 113) {            if (tmpArray.length >= 4 && tmpArray.length % 4 === 0) for (j = 0, k = tmpArray.length; j < k; j += 4) {              ctrlX = cx + tmpArray[j];              ctrlY = cy + tmpArray[j + 1];              endX = cx + tmpArray[j + 2];              endY = cy + tmpArray[j + 3];              this.parsePathQuadto(cx, cy, ctrlX, ctrlY, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 84) {            if (tmpArray.length >= 2 && tmpArray.length % 2 === 0) for (j = 0, k = tmpArray.length; j < k; j += 2) {              if (lastInstruction.toLowerCase() === "q" || lastInstruction.toLowerCase() === "t") {                ppx = this.vertices[this.vertices.length - 2][0];                ppy = this.vertices[this.vertices.length - 2][1];                px = this.vertices[this.vertices.length - 1][0];                py = this.vertices[this.vertices.length - 1][1];                ctrlX = px + (px - ppx);                ctrlY = py + (py - ppy)              } else {                ctrlX = cx;                ctrlY = cy              }              endX = tmpArray[j];              endY = tmpArray[j + 1];              this.parsePathQuadto(cx, cy, ctrlX, ctrlY, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 116) {            if (tmpArray.length >= 2 && tmpArray.length % 2 === 0) for (j = 0, k = tmpArray.length; j < k; j += 2) {              if (lastInstruction.toLowerCase() === "q" || lastInstruction.toLowerCase() === "t") {                ppx = this.vertices[this.vertices.length - 2][0];                ppy = this.vertices[this.vertices.length - 2][1];                px = this.vertices[this.vertices.length - 1][0];                py = this.vertices[this.vertices.length - 1][1];                ctrlX = px + (px - ppx);                ctrlY = py + (py - ppy)              } else {                ctrlX = cx;                ctrlY = cy              }              endX = cx + tmpArray[j];              endY = cy + tmpArray[j + 1];              this.parsePathQuadto(cx, cy, ctrlX, ctrlY, endX, endY);              cx = endX;              cy = endY            }          } else if (valOf === 90 || valOf === 122) this.close = true;          lastInstruction = command.toString()        } else i++      }    };    PShapeSVG.prototype.parsePathQuadto = function(x1, y1, cx, cy, x2, y2) {      if (this.vertices.length > 0) {        this.parsePathCode(1);        this.parsePathVertex(x1 + (cx - x1) * 2 / 3, y1 + (cy - y1) * 2 / 3);        this.parsePathVertex(x2 + (cx - x2) * 2 / 3, y2 + (cy - y2) * 2 / 3);        this.parsePathVertex(x2, y2)      } else throw "Path must start with M/m";    };    PShapeSVG.prototype.parsePathCurveto = function(x1, y1, x2, y2, x3, y3) {      if (this.vertices.length > 0) {        this.parsePathCode(1);        this.parsePathVertex(x1, y1);        this.parsePathVertex(x2, y2);        this.parsePathVertex(x3, y3)      } else throw "Path must start with M/m";    };    PShapeSVG.prototype.parsePathLineto = function(px, py) {      if (this.vertices.length > 0) {        this.parsePathCode(0);        this.parsePathVertex(px, py);        this.vertices[this.vertices.length - 1]["moveTo"] = false      } else throw "Path must start with M/m";    };    PShapeSVG.prototype.parsePathMoveto = function(px, py) {      if (this.vertices.length > 0) this.parsePathCode(3);      this.parsePathCode(0);      this.parsePathVertex(px, py);      this.vertices[this.vertices.length - 1]["moveTo"] = true    };    PShapeSVG.prototype.parsePathVertex = function(x, y) {      var verts = [];      verts[0] = x;      verts[1] = y;      this.vertices.push(verts)    };    PShapeSVG.prototype.parsePathCode = function(what) {      this.vertexCodes.push(what)    };    PShapeSVG.prototype.parsePoly = function(val) {      this.family = 21;      this.close = val;      var pointsAttr = p.trim(this.element.getStringAttribute("points").replace(/[,\s]+/g, " "));      if (pointsAttr !== null) {        var pointsBuffer = pointsAttr.split(" ");        if (pointsBuffer.length % 2 === 0) for (var i = 0, j = pointsBuffer.length; i < j; i++) {          var verts = [];          verts[0] = pointsBuffer[i];          verts[1] = pointsBuffer[++i];          this.vertices.push(verts)        } else throw "Error parsing polygon points: odd number of coordinates provided";      }    };    PShapeSVG.prototype.parseRect = function() {      this.kind = 30;      this.family = 1;      this.params = [];      this.params[0] = this.element.getFloatAttribute("x");      this.params[1] = this.element.getFloatAttribute("y");      this.params[2] = this.element.getFloatAttribute("width");      this.params[3] = this.element.getFloatAttribute("height");      if (this.params[2] < 0 || this.params[3] < 0) throw "svg error: negative width or height found while parsing <rect>";    };    PShapeSVG.prototype.parseEllipse = function(val) {      this.kind = 31;      this.family = 1;      this.params = [];      this.params[0] = this.element.getFloatAttribute("cx") | 0;      this.params[1] = this.element.getFloatAttribute("cy") | 0;      var rx, ry;      if (val) {        rx = ry = this.element.getFloatAttribute("r");        if (rx < 0) throw "svg error: negative radius found while parsing <circle>";      } else {        rx = this.element.getFloatAttribute("rx");        ry = this.element.getFloatAttribute("ry");        if (rx < 0 || ry < 0) throw "svg error: negative x-axis radius or y-axis radius found while parsing <ellipse>";      }      this.params[0] -= rx;      this.params[1] -= ry;      this.params[2] = rx * 2;      this.params[3] = ry * 2    };    PShapeSVG.prototype.parseLine = function() {      this.kind = 4;      this.family = 1;      this.params = [];      this.params[0] = this.element.getFloatAttribute("x1");      this.params[1] = this.element.getFloatAttribute("y1");      this.params[2] = this.element.getFloatAttribute("x2");      this.params[3] = this.element.getFloatAttribute("y2")    };    PShapeSVG.prototype.parseColors = function(element) {      if (element.hasAttribute("opacity")) this.setOpacity(element.getAttribute("opacity"));      if (element.hasAttribute("stroke")) this.setStroke(element.getAttribute("stroke"));      if (element.hasAttribute("stroke-width")) this.setStrokeWeight(element.getAttribute("stroke-width"));      if (element.hasAttribute("stroke-linejoin")) this.setStrokeJoin(element.getAttribute("stroke-linejoin"));      if (element.hasAttribute("stroke-linecap")) this.setStrokeCap(element.getStringAttribute("stroke-linecap"));      if (element.hasAttribute("fill")) this.setFill(element.getStringAttribute("fill"));      if (element.hasAttribute("style")) {        var styleText = element.getStringAttribute("style");        var styleTokens = styleText.toString().split(";");        for (var i = 0, j = styleTokens.length; i < j; i++) {          var tokens = p.trim(styleTokens[i].split(":"));          if (tokens[0] === "fill") this.setFill(tokens[1]);          else if (tokens[0] === "fill-opacity") this.setFillOpacity(tokens[1]);          else if (tokens[0] === "stroke") this.setStroke(tokens[1]);          else if (tokens[0] === "stroke-width") this.setStrokeWeight(tokens[1]);          else if (tokens[0] === "stroke-linecap") this.setStrokeCap(tokens[1]);          else if (tokens[0] === "stroke-linejoin") this.setStrokeJoin(tokens[1]);          else if (tokens[0] === "stroke-opacity") this.setStrokeOpacity(tokens[1]);          else if (tokens[0] === "opacity") this.setOpacity(tokens[1])        }      }    };    PShapeSVG.prototype.setFillOpacity = function(opacityText) {      this.fillOpacity = parseFloat(opacityText);      this.fillColor = this.fillOpacity * 255 << 24 | this.fillColor & 16777215    };    PShapeSVG.prototype.setFill = function(fillText) {      var opacityMask = this.fillColor & 4278190080;      if (fillText === "none") this.fill = false;      else if (fillText.indexOf("#") === 0) {        this.fill = true;        if (fillText.length === 4) fillText = fillText.replace(/#(.)(.)(.)/, "#$1$1$2$2$3$3");        this.fillColor = opacityMask | parseInt(fillText.substring(1), 16) & 16777215      } else if (fillText.indexOf("rgb") === 0) {        this.fill = true;        this.fillColor = opacityMask | this.parseRGB(fillText)      } else if (fillText.indexOf("url(#") === 0) this.fillName = fillText.substring(5, fillText.length - 1);      else if (colors[fillText]) {        this.fill = true;        this.fillColor = opacityMask | parseInt(colors[fillText].substring(1), 16) & 16777215      }    };    PShapeSVG.prototype.setOpacity = function(opacity) {      this.strokeColor = parseFloat(opacity) * 255 << 24 | this.strokeColor & 16777215;      this.fillColor = parseFloat(opacity) * 255 << 24 | this.fillColor & 16777215    };    PShapeSVG.prototype.setStroke = function(strokeText) {      var opacityMask = this.strokeColor & 4278190080;      if (strokeText === "none") this.stroke = false;      else if (strokeText.charAt(0) === "#") {        this.stroke = true;        if (strokeText.length === 4) strokeText = strokeText.replace(/#(.)(.)(.)/, "#$1$1$2$2$3$3");        this.strokeColor = opacityMask | parseInt(strokeText.substring(1), 16) & 16777215      } else if (strokeText.indexOf("rgb") === 0) {        this.stroke = true;        this.strokeColor = opacityMask | this.parseRGB(strokeText)      } else if (strokeText.indexOf("url(#") === 0) this.strokeName = strokeText.substring(5, strokeText.length - 1);      else if (colors[strokeText]) {        this.stroke = true;        this.strokeColor = opacityMask | parseInt(colors[strokeText].substring(1), 16) & 16777215      }    };    PShapeSVG.prototype.setStrokeWeight = function(weight) {      this.strokeWeight = this.parseUnitSize(weight)    };    PShapeSVG.prototype.setStrokeJoin = function(linejoin) {      if (linejoin === "miter") this.strokeJoin = 'miter';      else if (linejoin === "round") this.strokeJoin = 'round';      else if (linejoin === "bevel") this.strokeJoin = 'bevel'    };    PShapeSVG.prototype.setStrokeCap = function(linecap) {      if (linecap === "butt") this.strokeCap = 'butt';      else if (linecap === "round") this.strokeCap = 'round';      else if (linecap === "square") this.strokeCap = 'square'    };    PShapeSVG.prototype.setStrokeOpacity = function(opacityText) {      this.strokeOpacity = parseFloat(opacityText);      this.strokeColor = this.strokeOpacity * 255 << 24 | this.strokeColor & 16777215    };    PShapeSVG.prototype.parseRGB = function(color) {      var sub = color.substring(color.indexOf("(") + 1, color.indexOf(")"));      var values = sub.split(", ");      return values[0] << 16 | values[1] << 8 | values[2]    };    PShapeSVG.prototype.parseUnitSize = function(text) {      var len = text.length - 2;      if (len < 0) return text;      if (text.indexOf("pt") === len) return parseFloat(text.substring(0, len)) * 1.25;      if (text.indexOf("pc") === len) return parseFloat(text.substring(0, len)) * 15;      if (text.indexOf("mm") === len) return parseFloat(text.substring(0, len)) * 3.543307;      if (text.indexOf("cm") === len) return parseFloat(text.substring(0, len)) * 35.43307;      if (text.indexOf("in") === len) return parseFloat(text.substring(0, len)) * 90;      if (text.indexOf("px") === len) return parseFloat(text.substring(0, len));      return parseFloat(text)    };    p.shape = function(shape, x, y, width, height) {      if (arguments.length >= 1 && arguments[0] !== null) if (shape.isVisible()) {        p.pushMatrix();        if (curShapeMode === 3) if (arguments.length === 5) {          p.translate(x - width / 2, y - height / 2);          p.scale(width / shape.getWidth(), height / shape.getHeight())        } else if (arguments.length === 3) p.translate(x - shape.getWidth() / 2, -shape.getHeight() / 2);        else p.translate(-shape.getWidth() / 2, -shape.getHeight() / 2);        else if (curShapeMode === 0) if (arguments.length === 5) {          p.translate(x, y);          p.scale(width / shape.getWidth(), height / shape.getHeight())        } else {          if (arguments.length === 3) p.translate(x, y)        } else if (curShapeMode === 1) if (arguments.length === 5) {          width -= x;          height -= y;          p.translate(x, y);          p.scale(width / shape.getWidth(), height / shape.getHeight())        } else if (arguments.length === 3) p.translate(x, y);        shape.draw(p);        if (arguments.length === 1 && curShapeMode === 3 || arguments.length > 1) p.popMatrix()      }    };    p.shapeMode = function(mode) {      curShapeMode = mode    };    p.loadShape = function(filename) {      if (arguments.length === 1) if (filename.indexOf(".svg") > -1) return new PShapeSVG(null, filename);      return null    };    var XMLAttribute = function(fname, n, nameSpace, v, t) {      this.fullName = fname || "";      this.name = n || "";      this.namespace = nameSpace || "";      this.value = v;      this.type = t    };    XMLAttribute.prototype = {      getName: function() {        return this.name      },      getFullName: function() {        return this.fullName      },      getNamespace: function() {        return this.namespace      },      getValue: function() {        return this.value      },      getType: function() {        return this.type      },      setValue: function(newval) {        this.value = newval      }    };    var XMLElement = p.XMLElement = function(selector, uri, sysid, line) {      this.attributes = [];      this.children = [];      this.fullName = null;      this.name = null;      this.namespace = "";      this.content = null;      this.parent = null;      this.lineNr = "";      this.systemID = "";      this.type = "ELEMENT";      if (selector) if (typeof selector === "string") if (uri === undef && selector.indexOf("<") > -1) this.parse(selector);      else {        this.fullName = selector;        this.namespace = uri;        this.systemId = sysid;        this.lineNr = line      } else this.parse(uri)    };    XMLElement.prototype = {      parse: function(textstring) {        var xmlDoc;        try {          var extension = textstring.substring(textstring.length - 4);          if (extension === ".xml" || extension === ".svg") textstring = ajax(textstring);          xmlDoc = (new DOMParser).parseFromString(textstring, "text/xml");          var elements = xmlDoc.documentElement;          if (elements) this.parseChildrenRecursive(null, elements);          else throw "Error loading document";          return this        } catch(e) {          throw e;        }      },      parseChildrenRecursive: function(parent, elementpath) {        var xmlelement, xmlattribute, tmpattrib, l, m, child;        if (!parent) {          this.fullName = elementpath.localName;          this.name = elementpath.nodeName;          xmlelement = this        } else {          xmlelement = new XMLElement(elementpath.nodeName);          xmlelement.parent = parent        }        if (elementpath.nodeType === 3 && elementpath.textContent !== "") return this.createPCDataElement(elementpath.textContent);        if (elementpath.nodeType === 4) return this.createCDataElement(elementpath.textContent);        if (elementpath.attributes) for (l = 0, m = elementpath.attributes.length; l < m; l++) {          tmpattrib = elementpath.attributes[l];          xmlattribute = new XMLAttribute(tmpattrib.getname, tmpattrib.nodeName, tmpattrib.namespaceURI, tmpattrib.nodeValue, tmpattrib.nodeType);          xmlelement.attributes.push(xmlattribute)        }        if (elementpath.childNodes) for (l = 0, m = elementpath.childNodes.length; l < m; l++) {          var node = elementpath.childNodes[l];          child = xmlelement.parseChildrenRecursive(xmlelement, node);          if (child !== null) xmlelement.children.push(child)        }        return xmlelement      },      createElement: function(fullname, namespaceuri, sysid, line) {        if (sysid === undef) return new XMLElement(fullname, namespaceuri);        return new XMLElement(fullname, namespaceuri, sysid, line)      },      createPCDataElement: function(content, isCDATA) {        if (content.replace(/^\s+$/g, "") === "") return null;        var pcdata = new XMLElement;        pcdata.type = "TEXT";        pcdata.content = content;        return pcdata      },      createCDataElement: function(content) {        var cdata = this.createPCDataElement(content);        if (cdata === null) return null;        cdata.type = "CDATA";        var htmlentities = {          "<": "<",          ">": ">",          "'": "'",          '"': """        },          entity;        for (entity in htmlentities) if (!Object.hasOwnProperty(htmlentities, entity)) content = content.replace(new RegExp(entity, "g"), htmlentities[entity]);        cdata.cdata = content;        return cdata      },      hasAttribute: function() {        if (arguments.length === 1) return this.getAttribute(arguments[0]) !== null;        if (arguments.length === 2) return this.getAttribute(arguments[0], arguments[1]) !== null      },      equals: function(other) {        if (! (other instanceof XMLElement)) return false;        var i, j;        if (this.fullName !== other.fullName) return false;        if (this.attributes.length !== other.getAttributeCount()) return false;        if (this.attributes.length !== other.attributes.length) return false;        var attr_name, attr_ns, attr_value, attr_type, attr_other;        for (i = 0, j = this.attributes.length; i < j; i++) {          attr_name = this.attributes[i].getName();          attr_ns = this.attributes[i].getNamespace();          attr_other = other.findAttribute(attr_name, attr_ns);          if (attr_other === null) return false;          if (this.attributes[i].getValue() !== attr_other.getValue()) return false;          if (this.attributes[i].getType() !== attr_other.getType()) return false        }        if (this.children.length !== other.getChildCount()) return false;        if (this.children.length > 0) {          var child1, child2;          for (i = 0, j = this.children.length; i < j; i++) {            child1 = this.getChild(i);            child2 = other.getChild(i);            if (!child1.equals(child2)) return false          }          return true        }        return this.content === other.content      },      getContent: function() {        if (this.type === "TEXT" || this.type === "CDATA") return this.content;        var children = this.children;        if (children.length === 1 && (children[0].type === "TEXT" || children[0].type === "CDATA")) return children[0].content;        return null      },      getAttribute: function() {        var attribute;        if (arguments.length === 2) {          attribute = this.findAttribute(arguments[0]);          if (attribute) return attribute.getValue();          return arguments[1]        } else if (arguments.length === 1) {          attribute = this.findAttribute(arguments[0]);          if (attribute) return attribute.getValue();          return null        } else if (arguments.length === 3) {          attribute = this.findAttribute(arguments[0], arguments[1]);          if (attribute) return attribute.getValue();          return arguments[2]        }      },      getStringAttribute: function() {        if (arguments.length === 1) return this.getAttribute(arguments[0]);        if (arguments.length === 2) return this.getAttribute(arguments[0], arguments[1]);        return this.getAttribute(arguments[0], arguments[1], arguments[2])      },      getString: function(attributeName) {        return this.getStringAttribute(attributeName)      },      getFloatAttribute: function() {        if (arguments.length === 1) return parseFloat(this.getAttribute(arguments[0], 0));        if (arguments.length === 2) return this.getAttribute(arguments[0], arguments[1]);        return this.getAttribute(arguments[0], arguments[1], arguments[2])      },      getFloat: function(attributeName) {        return this.getFloatAttribute(attributeName)      },      getIntAttribute: function() {        if (arguments.length === 1) return this.getAttribute(arguments[0], 0);        if (arguments.length === 2) return this.getAttribute(arguments[0], arguments[1]);        return this.getAttribute(arguments[0], arguments[1], arguments[2])      },      getInt: function(attributeName) {        return this.getIntAttribute(attributeName)      },      hasChildren: function() {        return this.children.length > 0      },      addChild: function(child) {        if (child !== null) {          child.parent = this;          this.children.push(child)        }      },      insertChild: function(child, index) {        if (child) {          if (child.getLocalName() === null && !this.hasChildren()) {            var lastChild = this.children[this.children.length - 1];            if (lastChild.getLocalName() === null) {              lastChild.setContent(lastChild.getContent() + child.getContent());              return            }          }          child.parent = this;          this.children.splice(index, 0, child)        }      },      getChild: function(selector) {        if (typeof selector === "number") return this.children[selector];        if (selector.indexOf("/") !== -1) return this.getChildRecursive(selector.split("/"), 0);        var kid, kidName;        for (var i = 0, j = this.getChildCount(); i < j; i++) {          kid = this.getChild(i);          kidName = kid.getName();          if (kidName !== null && kidName === selector) return kid        }        return null      },      getChildren: function() {        if (arguments.length === 1) {          if (typeof arguments[0] === "number") return this.getChild(arguments[0]);          if (arguments[0].indexOf("/") !== -1) return this.getChildrenRecursive(arguments[0].split("/"), 0);          var matches = [];          var kid, kidName;          for (var i = 0, j = this.getChildCount(); i < j; i++) {            kid = this.getChild(i);            kidName = kid.getName();            if (kidName !== null && kidName === arguments[0]) matches.push(kid)          }          return matches        }        return this.children      },      getChildCount: function() {        return this.children.length      },      getChildRecursive: function(items, offset) {        if (offset === items.length) return this;        var kid, kidName, matchName = items[offset];        for (var i = 0, j = this.getChildCount(); i < j; i++) {          kid = this.getChild(i);          kidName = kid.getName();          if (kidName !== null && kidName === matchName) return kid.getChildRecursive(items, offset + 1)        }        return null      },      getChildrenRecursive: function(items, offset) {        if (offset === items.length - 1) return this.getChildren(items[offset]);        var matches = this.getChildren(items[offset]);        var kidMatches = [];        for (var i = 0; i < matches.length; i++) kidMatches = kidMatches.concat(matches[i].getChildrenRecursive(items, offset + 1));        return kidMatches      },      isLeaf: function() {        return !this.hasChildren()      },      listChildren: function() {        var arr = [];        for (var i = 0, j = this.children.length; i < j; i++) arr.push(this.getChild(i).getName());        return arr      },      removeAttribute: function(name, namespace) {        this.namespace = namespace || "";        for (var i = 0, j = this.attributes.length; i < j; i++) if (this.attributes[i].getName() === name && this.attributes[i].getNamespace() === this.namespace) {          this.attributes.splice(i, 1);          break        }      },      removeChild: function(child) {        if (child) for (var i = 0, j = this.children.length; i < j; i++) if (this.children[i].equals(child)) {          this.children.splice(i, 1);          break        }      },      removeChildAtIndex: function(index) {        if (this.children.length > index) this.children.splice(index, 1)      },      findAttribute: function(name, namespace) {        this.namespace = namespace || "";        for (var i = 0, j = this.attributes.length; i < j; i++) if (this.attributes[i].getName() === name && this.attributes[i].getNamespace() === this.namespace) return this.attributes[i];        return null      },      setAttribute: function() {        var attr;        if (arguments.length === 3) {          var index = arguments[0].indexOf(":");          var name = arguments[0].substring(index + 1);          attr = this.findAttribute(name, arguments[1]);          if (attr) attr.setValue(arguments[2]);          else {            attr = new XMLAttribute(arguments[0], name, arguments[1], arguments[2], "CDATA");            this.attributes.push(attr)          }        } else {          attr = this.findAttribute(arguments[0]);          if (attr) attr.setValue(arguments[1]);          else {            attr = new XMLAttribute(arguments[0], arguments[0], null, arguments[1], "CDATA");            this.attributes.push(attr)          }        }      },      setString: function(attribute, value) {        this.setAttribute(attribute, value)      },      setInt: function(attribute, value) {        this.setAttribute(attribute, value)      },      setFloat: function(attribute, value) {        this.setAttribute(attribute, value)      },      setContent: function(content) {        if (this.children.length > 0) Processing.debug("Tried to set content for XMLElement with children");        this.content = content      },      setName: function() {        if (arguments.length === 1) {          this.name = arguments[0];          this.fullName = arguments[0];          this.namespace = null        } else {          var index = arguments[0].indexOf(":");          if (arguments[1] === null || index < 0) this.name = arguments[0];          else this.name = arguments[0].substring(index + 1);          this.fullName = arguments[0];          this.namespace = arguments[1]        }      },      getName: function() {        return this.fullName      },      getLocalName: function() {        return this.name      },      getAttributeCount: function() {        return this.attributes.length      },      toString: function() {        if (this.type === "TEXT") return this.content;        if (this.type === "CDATA") return this.cdata;        var tagstring = this.fullName;        var xmlstring = "<" + tagstring;        var a, c;        for (a = 0; a < this.attributes.length; a++) {          var attr = this.attributes[a];          xmlstring += " " + attr.getName() + "=" + '"' + attr.getValue() + '"'        }        if (this.children.length === 0) if (this.content === "") xmlstring += "/>";        else xmlstring += ">" + this.content + "</" + tagstring + ">";        else {          xmlstring += ">";          for (c = 0; c < this.children.length; c++) xmlstring += this.children[c].toString();          xmlstring += "</" + tagstring + ">"        }        return xmlstring      }    };    XMLElement.parse = function(xmlstring) {      var element = new XMLElement;      element.parse(xmlstring);      return element    };    var XML = p.XML = p.XMLElement;    p.loadXML = function(uri) {      return new XML(p, uri)    };    var printMatrixHelper = function(elements) {      var big = 0;      for (var i = 0; i < elements.length; i++) if (i !== 0) big = Math.max(big, Math.abs(elements[i]));      else big = Math.abs(elements[i]);      var digits = (big + "").indexOf(".");      if (digits === 0) digits = 1;      else if (digits === -1) digits = (big + "").length;      return digits    };    var PMatrix2D = p.PMatrix2D = function() {      if (arguments.length === 0) this.reset();      else if (arguments.length === 1 && arguments[0] instanceof PMatrix2D) this.set(arguments[0].array());      else if (arguments.length === 6) this.set(arguments[0], arguments[1], arguments[2], arguments[3], arguments[4], arguments[5])    };    PMatrix2D.prototype = {      set: function() {        if (arguments.length === 6) {          var a = arguments;          this.set([a[0], a[1], a[2], a[3], a[4], a[5]])        } else if (arguments.length === 1 && arguments[0] instanceof PMatrix2D) this.elements = arguments[0].array();        else if (arguments.length === 1 && arguments[0] instanceof Array) this.elements = arguments[0].slice()      },      get: function() {        var outgoing = new PMatrix2D;        outgoing.set(this.elements);        return outgoing      },      reset: function() {        this.set([1, 0, 0, 0, 1, 0])      },      array: function array() {        return this.elements.slice()      },      translate: function(tx, ty) {        this.elements[2] = tx * this.elements[0] + ty * this.elements[1] + this.elements[2];        this.elements[5] = tx * this.elements[3] + ty * this.elements[4] + this.elements[5]      },      invTranslate: function(tx, ty) {        this.translate(-tx, -ty)      },      transpose: function() {},      mult: function(source, target) {        var x, y;        if (source instanceof        PVector) {          x = source.x;          y = source.y;          if (!target) target = new PVector        } else if (source instanceof Array) {          x = source[0];          y = source[1];          if (!target) target = []        }        if (target instanceof Array) {          target[0] = this.elements[0] * x + this.elements[1] * y + this.elements[2];          target[1] = this.elements[3] * x + this.elements[4] * y + this.elements[5]        } else if (target instanceof PVector) {          target.x = this.elements[0] * x + this.elements[1] * y + this.elements[2];          target.y = this.elements[3] * x + this.elements[4] * y + this.elements[5];          target.z = 0        }        return target      },      multX: function(x, y) {        return x * this.elements[0] + y * this.elements[1] + this.elements[2]      },      multY: function(x, y) {        return x * this.elements[3] + y * this.elements[4] + this.elements[5]      },      skewX: function(angle) {        this.apply(1, 0, 1, angle, 0, 0)      },      skewY: function(angle) {        this.apply(1, 0, 1, 0, angle, 0)      },      shearX: function(angle) {        this.apply(1, 0, 1, Math.tan(angle), 0, 0)      },      shearY: function(angle) {        this.apply(1, 0, 1, 0, Math.tan(angle), 0)      },      determinant: function() {        return this.elements[0] * this.elements[4] - this.elements[1] * this.elements[3]      },      invert: function() {        var d = this.determinant();        if (Math.abs(d) > -2147483648) {          var old00 = this.elements[0];          var old01 = this.elements[1];          var old02 = this.elements[2];          var old10 = this.elements[3];          var old11 = this.elements[4];          var old12 = this.elements[5];          this.elements[0] = old11 / d;          this.elements[3] = -old10 / d;          this.elements[1] = -old01 / d;          this.elements[4] = old00 / d;          this.elements[2] = (old01 * old12 - old11 * old02) / d;          this.elements[5] = (old10 * old02 - old00 * old12) / d;          return true        }        return false      },      scale: function(sx, sy) {        if (sx && !sy) sy = sx;        if (sx && sy) {          this.elements[0] *= sx;          this.elements[1] *= sy;          this.elements[3] *= sx;          this.elements[4] *= sy        }      },      invScale: function(sx, sy) {        if (sx && !sy) sy = sx;        this.scale(1 / sx, 1 / sy)      },      apply: function() {        var source;        if (arguments.length === 1 && arguments[0] instanceof PMatrix2D) source = arguments[0].array();        else if (arguments.length === 6) source = Array.prototype.slice.call(arguments);        else if (arguments.length === 1 && arguments[0] instanceof Array) source = arguments[0];        var result = [0, 0, this.elements[2], 0, 0, this.elements[5]];        var e = 0;        for (var row = 0; row < 2; row++) for (var col = 0; col < 3; col++, e++) result[e] += this.elements[row * 3 + 0] * source[col + 0] + this.elements[row * 3 + 1] * source[col + 3];        this.elements = result.slice()      },      preApply: function() {        var source;        if (arguments.length === 1 && arguments[0] instanceof PMatrix2D) source = arguments[0].array();        else if (arguments.length === 6) source = Array.prototype.slice.call(arguments);        else if (arguments.length === 1 && arguments[0] instanceof Array) source = arguments[0];        var result = [0, 0, source[2], 0, 0, source[5]];        result[2] = source[2] + this.elements[2] * source[0] + this.elements[5] * source[1];        result[5] = source[5] + this.elements[2] * source[3] + this.elements[5] * source[4];        result[0] = this.elements[0] * source[0] + this.elements[3] * source[1];        result[3] = this.elements[0] * source[3] + this.elements[3] * source[4];        result[1] = this.elements[1] * source[0] + this.elements[4] * source[1];        result[4] = this.elements[1] * source[3] + this.elements[4] * source[4];        this.elements = result.slice()      },      rotate: function(angle) {        var c = Math.cos(angle);        var s = Math.sin(angle);        var temp1 = this.elements[0];        var temp2 = this.elements[1];        this.elements[0] = c * temp1 + s * temp2;        this.elements[1] = -s * temp1 + c * temp2;        temp1 = this.elements[3];        temp2 = this.elements[4];        this.elements[3] = c * temp1 + s * temp2;        this.elements[4] = -s * temp1 + c * temp2      },      rotateZ: function(angle) {        this.rotate(angle)      },      invRotateZ: function(angle) {        this.rotateZ(angle - Math.PI)      },      print: function() {        var digits = printMatrixHelper(this.elements);        var output = "" + p.nfs(this.elements[0], digits, 4) + " " + p.nfs(this.elements[1], digits, 4) + " " + p.nfs(this.elements[2], digits, 4) + "\n" + p.nfs(this.elements[3], digits, 4) + " " + p.nfs(this.elements[4], digits, 4) + " " + p.nfs(this.elements[5], digits, 4) + "\n\n";        p.println(output)      }    };    var PMatrix3D = p.PMatrix3D = function() {      this.reset()    };    PMatrix3D.prototype = {      set: function() {        if (arguments.length === 16) this.elements = Array.prototype.slice.call(arguments);        else if (arguments.length === 1 && arguments[0] instanceof PMatrix3D) this.elements = arguments[0].array();        else if (arguments.length === 1 && arguments[0] instanceof Array) this.elements = arguments[0].slice()      },      get: function() {        var outgoing = new PMatrix3D;        outgoing.set(this.elements);        return outgoing      },      reset: function() {        this.elements = [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]      },      array: function array() {        return this.elements.slice()      },      translate: function(tx, ty, tz) {        if (tz === undef) tz = 0;        this.elements[3] += tx * this.elements[0] + ty * this.elements[1] + tz * this.elements[2];        this.elements[7] += tx * this.elements[4] + ty * this.elements[5] + tz * this.elements[6];        this.elements[11] += tx * this.elements[8] + ty * this.elements[9] + tz * this.elements[10];        this.elements[15] += tx * this.elements[12] + ty * this.elements[13] + tz * this.elements[14]      },      transpose: function() {        var temp = this.elements[4];        this.elements[4] = this.elements[1];        this.elements[1] = temp;        temp = this.elements[8];        this.elements[8] = this.elements[2];        this.elements[2] = temp;        temp = this.elements[6];        this.elements[6] = this.elements[9];        this.elements[9] = temp;        temp = this.elements[3];        this.elements[3] = this.elements[12];        this.elements[12] = temp;        temp = this.elements[7];        this.elements[7] = this.elements[13];        this.elements[13] = temp;        temp = this.elements[11];        this.elements[11] = this.elements[14];        this.elements[14] = temp      },      mult: function(source, target) {        var x, y, z, w;        if (source instanceof        PVector) {          x = source.x;          y = source.y;          z = source.z;          w = 1;          if (!target) target = new PVector        } else if (source instanceof Array) {          x = source[0];          y = source[1];          z = source[2];          w = source[3] || 1;          if (!target || target.length !== 3 && target.length !== 4) target = [0, 0, 0]        }        if (target instanceof Array) if (target.length === 3) {          target[0] = this.elements[0] * x + this.elements[1] * y + this.elements[2] * z + this.elements[3];          target[1] = this.elements[4] * x + this.elements[5] * y + this.elements[6] * z + this.elements[7];          target[2] = this.elements[8] * x + this.elements[9] * y + this.elements[10] * z + this.elements[11]        } else if (target.length === 4) {          target[0] = this.elements[0] * x + this.elements[1] * y + this.elements[2] * z + this.elements[3] * w;          target[1] = this.elements[4] * x + this.elements[5] * y + this.elements[6] * z + this.elements[7] * w;          target[2] = this.elements[8] * x + this.elements[9] * y + this.elements[10] * z + this.elements[11] * w;          target[3] = this.elements[12] * x + this.elements[13] * y + this.elements[14] * z + this.elements[15] * w        }        if (target instanceof PVector) {          target.x = this.elements[0] * x + this.elements[1] * y + this.elements[2] * z + this.elements[3];          target.y = this.elements[4] * x + this.elements[5] * y + this.elements[6] * z + this.elements[7];          target.z = this.elements[8] * x + this.elements[9] * y + this.elements[10] * z + this.elements[11]        }        return target      },      preApply: function() {        var source;        if (arguments.length === 1 && arguments[0] instanceof PMatrix3D) source = arguments[0].array();        else if (arguments.length === 16) source = Array.prototype.slice.call(arguments);        else if (arguments.length === 1 && arguments[0] instanceof Array) source = arguments[0];        var result = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];        var e = 0;        for (var row = 0; row < 4; row++) for (var col = 0; col < 4; col++, e++) result[e] += this.elements[col + 0] * source[row * 4 + 0] + this.elements[col + 4] * source[row * 4 + 1] + this.elements[col + 8] * source[row * 4 + 2] + this.elements[col + 12] * source[row * 4 + 3];        this.elements = result.slice()      },      apply: function() {        var source;        if (arguments.length === 1 && arguments[0] instanceof PMatrix3D) source = arguments[0].array();        else if (arguments.length === 16) source = Array.prototype.slice.call(arguments);        else if (arguments.length === 1 && arguments[0] instanceof Array) source = arguments[0];        var result = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];        var e = 0;        for (var row = 0; row < 4; row++) for (var col = 0; col < 4; col++, e++) result[e] += this.elements[row * 4 + 0] * source[col + 0] + this.elements[row * 4 + 1] * source[col + 4] + this.elements[row * 4 + 2] * source[col + 8] + this.elements[row * 4 + 3] * source[col + 12];        this.elements = result.slice()      },      rotate: function(angle, v0, v1, v2) {        if (!v1) this.rotateZ(angle);        else {          var c = p.cos(angle);          var s = p.sin(angle);          var t = 1 - c;          this.apply(t * v0 * v0 + c, t * v0 * v1 - s * v2, t * v0 * v2 + s * v1, 0, t * v0 * v1 + s * v2, t * v1 * v1 + c, t * v1 * v2 - s * v0, 0, t * v0 * v2 - s * v1, t * v1 * v2 + s * v0, t * v2 * v2 + c, 0, 0, 0, 0, 1)        }      },      invApply: function() {        if (inverseCopy === undef) inverseCopy = new PMatrix3D;        var a = arguments;        inverseCopy.set(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]);        if (!inverseCopy.invert()) return false;        this.preApply(inverseCopy);        return true      },      rotateX: function(angle) {        var c = p.cos(angle);        var s = p.sin(angle);        this.apply([1, 0, 0, 0, 0, c, -s, 0, 0, s, c, 0, 0, 0, 0, 1])      },      rotateY: function(angle) {        var c = p.cos(angle);        var s = p.sin(angle);        this.apply([c,          0, s, 0, 0, 1, 0, 0, -s, 0, c, 0, 0, 0, 0, 1])      },      rotateZ: function(angle) {        var c = Math.cos(angle);        var s = Math.sin(angle);        this.apply([c, -s, 0, 0, s, c, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1])      },      scale: function(sx, sy, sz) {        if (sx && !sy && !sz) sy = sz = sx;        else if (sx && sy && !sz) sz = 1;        if (sx && sy && sz) {          this.elements[0] *= sx;          this.elements[1] *= sy;          this.elements[2] *= sz;          this.elements[4] *= sx;          this.elements[5] *= sy;          this.elements[6] *= sz;          this.elements[8] *= sx;          this.elements[9] *= sy;          this.elements[10] *= sz;          this.elements[12] *= sx;          this.elements[13] *= sy;          this.elements[14] *= sz        }      },      skewX: function(angle) {        var t = Math.tan(angle);        this.apply(1, t, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1)      },      skewY: function(angle) {        var t = Math.tan(angle);        this.apply(1, 0, 0, 0, t, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1)      },      shearX: function(angle) {        var t = Math.tan(angle);        this.apply(1, t, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1)      },      shearY: function(angle) {        var t = Math.tan(angle);        this.apply(1, 0, 0, 0, t, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1)      },      multX: function(x, y, z, w) {        if (!z) return this.elements[0] * x + this.elements[1] * y + this.elements[3];        if (!w) return this.elements[0] * x + this.elements[1] * y + this.elements[2] * z + this.elements[3];        return this.elements[0] * x + this.elements[1] * y + this.elements[2] * z + this.elements[3] * w      },      multY: function(x, y, z, w) {        if (!z) return this.elements[4] * x + this.elements[5] * y + this.elements[7];        if (!w) return this.elements[4] * x + this.elements[5] * y + this.elements[6] * z + this.elements[7];        return this.elements[4] * x + this.elements[5] * y + this.elements[6] * z + this.elements[7] * w      },      multZ: function(x, y, z, w) {        if (!w) return this.elements[8] * x + this.elements[9] * y + this.elements[10] * z + this.elements[11];        return this.elements[8] * x + this.elements[9] * y + this.elements[10] * z + this.elements[11] * w      },      multW: function(x, y, z, w) {        if (!w) return this.elements[12] * x + this.elements[13] * y + this.elements[14] * z + this.elements[15];        return this.elements[12] * x + this.elements[13] * y + this.elements[14] * z + this.elements[15] * w      },      invert: function() {        var fA0 = this.elements[0] * this.elements[5] - this.elements[1] * this.elements[4];        var fA1 = this.elements[0] * this.elements[6] - this.elements[2] * this.elements[4];        var fA2 = this.elements[0] * this.elements[7] - this.elements[3] * this.elements[4];        var fA3 = this.elements[1] * this.elements[6] - this.elements[2] * this.elements[5];        var fA4 = this.elements[1] * this.elements[7] - this.elements[3] * this.elements[5];        var fA5 = this.elements[2] * this.elements[7] - this.elements[3] * this.elements[6];        var fB0 = this.elements[8] * this.elements[13] - this.elements[9] * this.elements[12];        var fB1 = this.elements[8] * this.elements[14] - this.elements[10] * this.elements[12];        var fB2 = this.elements[8] * this.elements[15] - this.elements[11] * this.elements[12];        var fB3 = this.elements[9] * this.elements[14] - this.elements[10] * this.elements[13];        var fB4 = this.elements[9] * this.elements[15] - this.elements[11] * this.elements[13];        var fB5 = this.elements[10] * this.elements[15] - this.elements[11] * this.elements[14];        var fDet = fA0 * fB5 - fA1 * fB4 + fA2 * fB3 + fA3 * fB2 - fA4 * fB1 + fA5 * fB0;        if (Math.abs(fDet) <= 1.0E-9) return false;        var kInv = [];        kInv[0] = +this.elements[5] * fB5 - this.elements[6] * fB4 + this.elements[7] * fB3;        kInv[4] = -this.elements[4] * fB5 + this.elements[6] * fB2 - this.elements[7] * fB1;        kInv[8] = +this.elements[4] * fB4 - this.elements[5] * fB2 + this.elements[7] * fB0;        kInv[12] = -this.elements[4] * fB3 + this.elements[5] * fB1 - this.elements[6] * fB0;        kInv[1] = -this.elements[1] * fB5 + this.elements[2] * fB4 - this.elements[3] * fB3;        kInv[5] = +this.elements[0] * fB5 - this.elements[2] * fB2 + this.elements[3] * fB1;        kInv[9] = -this.elements[0] * fB4 + this.elements[1] * fB2 - this.elements[3] * fB0;        kInv[13] = +this.elements[0] * fB3 - this.elements[1] * fB1 + this.elements[2] * fB0;        kInv[2] = +this.elements[13] * fA5 - this.elements[14] * fA4 + this.elements[15] * fA3;        kInv[6] = -this.elements[12] * fA5 + this.elements[14] * fA2 - this.elements[15] * fA1;        kInv[10] = +this.elements[12] * fA4 - this.elements[13] * fA2 + this.elements[15] * fA0;        kInv[14] = -this.elements[12] * fA3 + this.elements[13] * fA1 - this.elements[14] * fA0;        kInv[3] = -this.elements[9] * fA5 + this.elements[10] * fA4 - this.elements[11] * fA3;        kInv[7] = +this.elements[8] * fA5 - this.elements[10] * fA2 + this.elements[11] * fA1;        kInv[11] = -this.elements[8] * fA4 + this.elements[9] * fA2 - this.elements[11] * fA0;        kInv[15] = +this.elements[8] * fA3 - this.elements[9] * fA1 + this.elements[10] * fA0;        var fInvDet = 1 / fDet;        kInv[0] *= fInvDet;        kInv[1] *= fInvDet;        kInv[2] *= fInvDet;        kInv[3] *= fInvDet;        kInv[4] *= fInvDet;        kInv[5] *= fInvDet;        kInv[6] *= fInvDet;        kInv[7] *= fInvDet;        kInv[8] *= fInvDet;        kInv[9] *= fInvDet;        kInv[10] *= fInvDet;        kInv[11] *= fInvDet;        kInv[12] *= fInvDet;        kInv[13] *= fInvDet;        kInv[14] *= fInvDet;        kInv[15] *= fInvDet;        this.elements = kInv.slice();        return true      },      toString: function() {        var str = "";        for (var i = 0; i < 15; i++) str += this.elements[i] + ", ";        str += this.elements[15];        return str      },      print: function() {        var digits = printMatrixHelper(this.elements);        var output = "" + p.nfs(this.elements[0], digits, 4) + " " + p.nfs(this.elements[1], digits, 4) + " " + p.nfs(this.elements[2], digits, 4) + " " + p.nfs(this.elements[3], digits, 4) + "\n" + p.nfs(this.elements[4], digits, 4) + " " + p.nfs(this.elements[5], digits, 4) + " " + p.nfs(this.elements[6], digits, 4) + " " + p.nfs(this.elements[7], digits, 4) + "\n" + p.nfs(this.elements[8], digits, 4) + " " + p.nfs(this.elements[9], digits, 4) + " " + p.nfs(this.elements[10], digits, 4) + " " + p.nfs(this.elements[11], digits, 4) + "\n" + p.nfs(this.elements[12], digits, 4) + " " + p.nfs(this.elements[13], digits, 4) + " " + p.nfs(this.elements[14], digits, 4) + " " + p.nfs(this.elements[15], digits, 4) + "\n\n";        p.println(output)      },      invTranslate: function(tx, ty, tz) {        this.preApply(1, 0, 0, -tx, 0, 1, 0, -ty, 0, 0, 1, -tz, 0, 0, 0, 1)      },      invRotateX: function(angle) {        var c = Math.cos(-angle);        var s = Math.sin(-angle);        this.preApply([1, 0, 0, 0, 0, c, -s, 0, 0, s, c, 0, 0, 0, 0, 1])      },      invRotateY: function(angle) {        var c = Math.cos(-angle);        var s = Math.sin(-angle);        this.preApply([c, 0, s, 0, 0, 1, 0, 0, -s, 0, c, 0, 0, 0, 0, 1])      },      invRotateZ: function(angle) {        var c = Math.cos(-angle);        var s = Math.sin(-angle);        this.preApply([c, -s, 0, 0, s, c, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1])      },      invScale: function(x, y, z) {        this.preApply([1 / x, 0, 0, 0, 0, 1 / y, 0, 0, 0, 0, 1 / z, 0, 0, 0, 0, 1])      }    };    var PMatrixStack = p.PMatrixStack = function() {      this.matrixStack = []    };    PMatrixStack.prototype.load = function() {      var tmpMatrix = drawing.$newPMatrix();      if (arguments.length === 1) tmpMatrix.set(arguments[0]);      else tmpMatrix.set(arguments);      this.matrixStack.push(tmpMatrix)    };    Drawing2D.prototype.$newPMatrix = function() {      return new PMatrix2D    };    Drawing3D.prototype.$newPMatrix = function() {      return new PMatrix3D    };    PMatrixStack.prototype.push = function() {      this.matrixStack.push(this.peek())    };    PMatrixStack.prototype.pop = function() {      return this.matrixStack.pop()    };    PMatrixStack.prototype.peek = function() {      var tmpMatrix = drawing.$newPMatrix();      tmpMatrix.set(this.matrixStack[this.matrixStack.length - 1]);      return tmpMatrix    };    PMatrixStack.prototype.mult = function(matrix) {      this.matrixStack[this.matrixStack.length - 1].apply(matrix)    };    p.split = function(str, delim) {      return str.split(delim)    };    p.splitTokens = function(str, tokens) {      if (tokens === undef) return str.split(/\s+/g);      var chars = tokens.split(/()/g),        buffer = "",        len = str.length,        i, c, tokenized = [];      for (i = 0; i < len; i++) {        c = str[i];        if (chars.indexOf(c) > -1) {          if (buffer !== "") tokenized.push(buffer);          buffer = ""        } else buffer += c      }      if (buffer !== "") tokenized.push(buffer);      return tokenized    };    p.append = function(array, element) {      array[array.length] = element;      return array    };    p.concat = function(array1, array2) {      return array1.concat(array2)    };    p.sort = function(array, numElem) {      var ret = [];      if (array.length > 0) {        var elemsToCopy = numElem > 0 ? numElem : array.length;        for (var i = 0; i < elemsToCopy; i++) ret.push(array[i]);        if (typeof array[0] === "string") ret.sort();        else ret.sort(function(a, b) {          return a - b        });        if (numElem > 0) for (var j = ret.length; j < array.length; j++) ret.push(array[j])      }      return ret    };    p.splice = function(array, value, index) {      if (value.length === 0) return array;      if (value instanceof Array) for (var i = 0, j = index; i < value.length; j++, i++) array.splice(j, 0, value[i]);      else array.splice(index, 0, value);      return array    };    p.subset = function(array, offset, length) {      var end = length !== undef ? offset + length : array.length;      return array.slice(offset, end)    };    p.join = function(array, seperator) {      return array.join(seperator)    };    p.shorten = function(ary) {      var newary = [];      var len = ary.length;      for (var i = 0; i < len; i++) newary[i] = ary[i];      newary.pop();      return newary    };    p.expand = function(ary, targetSize) {      var temp = ary.slice(0),        newSize = targetSize || ary.length * 2;      temp.length = newSize;      return temp    };    p.arrayCopy = function() {      var src, srcPos = 0,        dest, destPos = 0,        length;      if (arguments.length === 2) {        src = arguments[0];        dest = arguments[1];        length = src.length      } else if (arguments.length === 3) {        src = arguments[0];        dest = arguments[1];        length = arguments[2]      } else if (arguments.length === 5) {        src = arguments[0];        srcPos = arguments[1];        dest = arguments[2];        destPos = arguments[3];        length = arguments[4]      }      for (var i = srcPos, j = destPos; i < length + srcPos; i++, j++) if (dest[j] !== undef) dest[j] = src[i];      else throw "array index out of bounds exception";    };    p.reverse = function(array) {      return array.reverse()    };    p.mix = function(a, b, f) {      return a + ((b - a) * f >> 8)    };    p.peg = function(n) {      return n < 0 ? 0 : n > 255 ? 255 : n    };    p.modes = function() {      var ALPHA_MASK = 4278190080,        RED_MASK = 16711680,        GREEN_MASK = 65280,        BLUE_MASK = 255,        min = Math.min,        max = Math.max;      function applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb) {        var a = min(((c1 & 4278190080) >>> 24) + f, 255) << 24;        var r = ar + ((cr - ar) * f >> 8);        r = (r < 0 ? 0 : r > 255 ? 255 : r) << 16;        var g = ag + ((cg - ag) * f >> 8);        g = (g < 0 ? 0 : g > 255 ? 255 : g) << 8;        var b = ab + ((cb - ab) * f >> 8);        b = b < 0 ? 0 : b > 255 ? 255 : b;        return a | r | g | b      }      return {        replace: function(c1, c2) {          return c2        },        blend: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = c1 & RED_MASK,            ag = c1 & GREEN_MASK,            ab = c1 & BLUE_MASK,            br = c2 & RED_MASK,            bg = c2 & GREEN_MASK,            bb = c2 & BLUE_MASK;          return min(((c1 & ALPHA_MASK) >>> 24) + f, 255) << 24 | ar + ((br - ar) * f >> 8) & RED_MASK | ag + ((bg - ag) * f >> 8) & GREEN_MASK | ab + ((bb - ab) * f >> 8) & BLUE_MASK        },        add: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24;          return min(((c1 & ALPHA_MASK) >>> 24) + f, 255) << 24 | min((c1 & RED_MASK) + ((c2 & RED_MASK) >> 8) * f, RED_MASK) & RED_MASK | min((c1 & GREEN_MASK) + ((c2 & GREEN_MASK) >> 8) * f, GREEN_MASK) & GREEN_MASK | min((c1 & BLUE_MASK) + ((c2 & BLUE_MASK) * f >> 8), BLUE_MASK)        },        subtract: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24;          return min(((c1 & ALPHA_MASK) >>> 24) + f, 255) << 24 | max((c1 & RED_MASK) - ((c2 & RED_MASK) >> 8) * f, GREEN_MASK) & RED_MASK | max((c1 & GREEN_MASK) - ((c2 & GREEN_MASK) >> 8) * f, BLUE_MASK) & GREEN_MASK | max((c1 & BLUE_MASK) - ((c2 & BLUE_MASK) * f >> 8), 0)        },        lightest: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24;          return min(((c1 & ALPHA_MASK) >>> 24) + f, 255) << 24 | max(c1 & RED_MASK, ((c2 & RED_MASK) >> 8) * f) & RED_MASK | max(c1 & GREEN_MASK, ((c2 & GREEN_MASK) >> 8) * f) & GREEN_MASK | max(c1 & BLUE_MASK, (c2 & BLUE_MASK) * f >> 8)        },        darkest: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = c1 & RED_MASK,            ag = c1 & GREEN_MASK,            ab = c1 & BLUE_MASK,            br = min(c1 & RED_MASK, ((c2 & RED_MASK) >> 8) * f),            bg = min(c1 & GREEN_MASK, ((c2 & GREEN_MASK) >> 8) * f),            bb = min(c1 & BLUE_MASK, (c2 & BLUE_MASK) * f >> 8);          return min(((c1 & ALPHA_MASK) >>> 24) + f, 255) << 24 | ar + ((br - ar) * f >> 8) & RED_MASK | ag + ((bg - ag) * f >> 8) & GREEN_MASK | ab + ((bb - ab) * f >> 8) & BLUE_MASK        },        difference: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = ar > br ? ar - br : br - ar,          cg = ag > bg ? ag - bg : bg - ag,          cb = ab > bb ? ab - bb : bb - ab;          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        exclusion: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = ar + br - (ar * br >> 7),            cg = ag + bg - (ag * bg >> 7),            cb = ab + bb - (ab * bb >> 7);          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        multiply: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = ar * br >> 8,            cg = ag * bg >> 8,            cb = ab * bb >> 8;          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        screen: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = 255 - ((255 - ar) * (255 - br) >> 8),            cg = 255 - ((255 - ag) * (255 - bg) >> 8),            cb = 255 - ((255 - ab) * (255 - bb) >> 8);          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        hard_light: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = br < 128 ? ar * br >> 7 : 255 - ((255 - ar) * (255 - br) >> 7),          cg = bg < 128 ? ag * bg >> 7 : 255 - ((255 - ag) * (255 - bg) >> 7),          cb = bb < 128 ? ab * bb >> 7 : 255 - ((255 - ab) * (255 - bb) >> 7);          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        soft_light: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = (ar * br >> 7) + (ar * ar >> 8) - (ar * ar * br >> 15),            cg = (ag * bg >> 7) + (ag * ag >> 8) - (ag * ag * bg >> 15),            cb = (ab * bb >> 7) + (ab * ab >> 8) - (ab * ab * bb >> 15);          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        overlay: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK,            cr = ar < 128 ? ar * br >> 7 : 255 - ((255 - ar) * (255 - br) >> 7),          cg = ag < 128 ? ag * bg >> 7 : 255 - ((255 - ag) * (255 - bg) >> 7),          cb = ab < 128 ? ab * bb >> 7 : 255 - ((255 - ab) * (255 - bb) >> 7);          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        dodge: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK;          var cr = 255;          if (br !== 255) {            cr = (ar << 8) / (255 - br);            cr = cr < 0 ? 0 : cr > 255 ? 255 : cr          }          var cg = 255;          if (bg !== 255) {            cg = (ag << 8) / (255 - bg);            cg = cg < 0 ? 0 : cg > 255 ? 255 : cg          }          var cb = 255;          if (bb !== 255) {            cb = (ab << 8) / (255 - bb);            cb = cb < 0 ? 0 : cb > 255 ? 255 : cb          }          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        },        burn: function(c1, c2) {          var f = (c2 & ALPHA_MASK) >>> 24,            ar = (c1 & RED_MASK) >> 16,            ag = (c1 & GREEN_MASK) >> 8,            ab = c1 & BLUE_MASK,            br = (c2 & RED_MASK) >> 16,            bg = (c2 & GREEN_MASK) >> 8,            bb = c2 & BLUE_MASK;          var cr = 0;          if (br !== 0) {            cr = (255 - ar << 8) / br;            cr = 255 - (cr < 0 ? 0 : cr > 255 ? 255 : cr)          }          var cg = 0;          if (bg !== 0) {            cg = (255 - ag << 8) / bg;            cg = 255 - (cg < 0 ? 0 : cg > 255 ? 255 : cg)          }          var cb = 0;          if (bb !== 0) {            cb = (255 - ab << 8) / bb;            cb = 255 - (cb < 0 ? 0 : cb > 255 ? 255 : cb)          }          return applyMode(c1, f, ar, ag, ab, br, bg, bb, cr, cg, cb)        }      }    }();    function color$4(aValue1, aValue2, aValue3, aValue4) {      var r, g, b, a;      if (curColorMode === 3) {        var rgb = p.color.toRGB(aValue1, aValue2, aValue3);        r = rgb[0];        g = rgb[1];        b = rgb[2]      } else {        r = Math.round(255 * (aValue1 / colorModeX));        g = Math.round(255 * (aValue2 / colorModeY));        b = Math.round(255 * (aValue3 / colorModeZ))      }      a = Math.round(255 * (aValue4 / colorModeA));      r = r < 0 ? 0 : r;      g = g < 0 ? 0 : g;      b = b < 0 ? 0 : b;      a = a < 0 ? 0 : a;      r = r > 255 ? 255 : r;      g = g > 255 ? 255 : g;      b = b > 255 ? 255 : b;      a = a > 255 ? 255 : a;      return a << 24 & 4278190080 | r << 16 & 16711680 | g << 8 & 65280 | b & 255    }    function color$2(aValue1, aValue2) {      var a;      if (aValue1 & 4278190080) {        a = Math.round(255 * (aValue2 / colorModeA));        a = a > 255 ? 255 : a;        a = a < 0 ? 0 : a;        return aValue1 - (aValue1 & 4278190080) + (a << 24 & 4278190080)      }      if (curColorMode === 1) return color$4(aValue1, aValue1, aValue1, aValue2);      if (curColorMode === 3) return color$4(0, 0, aValue1 / colorModeX * colorModeZ, aValue2)    }    function color$1(aValue1) {      if (aValue1 <= colorModeX && aValue1 >= 0) {        if (curColorMode === 1) return color$4(aValue1, aValue1, aValue1, colorModeA);        if (curColorMode === 3) return color$4(0, 0, aValue1 / colorModeX * colorModeZ, colorModeA)      }      if (aValue1) {        if (aValue1 > 2147483647) aValue1 -= 4294967296;        return aValue1      }    }    p.color = function(aValue1, aValue2, aValue3, aValue4) {      if (aValue1 !== undef && aValue2 !== undef && aValue3 !== undef && aValue4 !== undef) return color$4(aValue1, aValue2, aValue3, aValue4);      if (aValue1 !== undef && aValue2 !== undef && aValue3 !== undef) return color$4(aValue1, aValue2, aValue3, colorModeA);      if (aValue1 !== undef && aValue2 !== undef) return color$2(aValue1, aValue2);      if (typeof aValue1 === "number") return color$1(aValue1);      return color$4(colorModeX, colorModeY, colorModeZ, colorModeA)    };    p.color.toString = function(colorInt) {      return "rgba(" + ((colorInt >> 16) & 255) + "," + ((colorInt >> 8) & 255) + "," + (colorInt & 255) + "," + ((colorInt >> 24) & 255) / 255 + ")"    };    p.color.toInt = function(r, g, b, a) {      return a << 24 & 4278190080 | r << 16 & 16711680 | g << 8 & 65280 | b & 255    };    p.color.toArray = function(colorInt) {      return [(colorInt >> 16) & 255, (colorInt >> 8) & 255, colorInt & 255, (colorInt >> 24) & 255]    };    p.color.toGLArray = function(colorInt) {      return [((colorInt & 16711680) >>> 16) / 255, ((colorInt >> 8) & 255) / 255, (colorInt & 255) / 255, ((colorInt >> 24) & 255) / 255]    };    p.color.toRGB = function(h, s, b) {      h = h > colorModeX ? colorModeX : h;      s = s > colorModeY ? colorModeY : s;      b = b > colorModeZ ? colorModeZ : b;      h = h / colorModeX * 360;      s = s / colorModeY * 100;      b = b / colorModeZ * 100;      var br = Math.round(b / 100 * 255);      if (s === 0) return [br, br, br];      var hue = h % 360;      var f = hue % 60;      var p = Math.round(b * (100 - s) / 1E4 * 255);      var q = Math.round(b * (6E3 - s * f) / 6E5 * 255);      var t = Math.round(b * (6E3 - s * (60 - f)) / 6E5 * 255);      switch (Math.floor(hue / 60)) {      case 0:        return [br, t, p];      case 1:        return [q, br, p];      case 2:        return [p, br, t];      case 3:        return [p, q, br];      case 4:        return [t, p, br];      case 5:        return [br, p, q]      }    };    function colorToHSB(colorInt) {      var red, green, blue;      red = ((colorInt >> 16) & 255) / 255;      green = ((colorInt >> 8) & 255) / 255;      blue = (colorInt & 255) / 255;      var max = p.max(p.max(red, green), blue),        min = p.min(p.min(red, green), blue),        hue, saturation;      if (min === max) return [0, 0, max * colorModeZ];      saturation = (max - min) / max;      if (red === max) hue = (green - blue) / (max - min);      else if (green === max) hue = 2 + (blue - red) / (max - min);      else hue = 4 + (red - green) / (max - min);      hue /= 6;      if (hue < 0) hue += 1;      else if (hue > 1) hue -= 1;      return [hue * colorModeX, saturation * colorModeY, max * colorModeZ]    }    p.brightness = function(colInt) {      return colorToHSB(colInt)[2]    };    p.saturation = function(colInt) {      return colorToHSB(colInt)[1]    };    p.hue = function(colInt) {      return colorToHSB(colInt)[0]    };    p.red = function(aColor) {      return ((aColor >> 16) & 255) / 255 * colorModeX    };    p.green = function(aColor) {      return ((aColor & 65280) >>> 8) / 255 * colorModeY    };    p.blue = function(aColor) {      return (aColor & 255) / 255 * colorModeZ    };    p.alpha = function(aColor) {      return ((aColor >> 24) & 255) / 255 * colorModeA    };    p.lerpColor = function(c1, c2, amt) {      var r, g, b, a, r1, g1, b1, a1, r2, g2, b2, a2;      var hsb1, hsb2, rgb, h, s;      var colorBits1 = p.color(c1);      var colorBits2 = p.color(c2);      if (curColorMode === 3) {        hsb1 = colorToHSB(colorBits1);        a1 = ((colorBits1 >> 24) & 255) / colorModeA;        hsb2 = colorToHSB(colorBits2);        a2 = ((colorBits2 & 4278190080) >>> 24) / colorModeA;        h = p.lerp(hsb1[0], hsb2[0], amt);        s = p.lerp(hsb1[1], hsb2[1], amt);        b = p.lerp(hsb1[2], hsb2[2], amt);        rgb = p.color.toRGB(h, s, b);        a = p.lerp(a1, a2, amt) * colorModeA;        return a << 24 & 4278190080 | (rgb[0] & 255) << 16 | (rgb[1] & 255) << 8 | rgb[2] & 255      }      r1 = (colorBits1 >> 16) & 255;      g1 = (colorBits1 >> 8) & 255;      b1 = colorBits1 & 255;      a1 = ((colorBits1 >> 24) & 255) / colorModeA;      r2 = (colorBits2 & 16711680) >>> 16;      g2 = (colorBits2 >> 8) & 255;      b2 = colorBits2 & 255;      a2 = ((colorBits2 >> 24) & 255) / colorModeA;      r = p.lerp(r1, r2, amt) | 0;      g = p.lerp(g1, g2, amt) | 0;      b = p.lerp(b1, b2, amt) | 0;      a = p.lerp(a1, a2, amt) * colorModeA;      return a << 24 & 4278190080 | r << 16 & 16711680 | g << 8 & 65280 | b & 255    };    p.colorMode = function() {      curColorMode = arguments[0];      if (arguments.length > 1) {        colorModeX = arguments[1];        colorModeY = arguments[2] || arguments[1];        colorModeZ = arguments[3] || arguments[1];        colorModeA = arguments[4] || arguments[1]      }    };    p.blendColor = function(c1, c2, mode) {      if (mode === 0) return p.modes.replace(c1, c2);      else if (mode === 1) return p.modes.blend(c1, c2);      else if (mode === 2) return p.modes.add(c1, c2);      else if (mode === 4) return p.modes.subtract(c1, c2);      else if (mode === 8) return p.modes.lightest(c1, c2);      else if (mode === 16) return p.modes.darkest(c1, c2);      else if (mode === 32) return p.modes.difference(c1, c2);      else if (mode === 64) return p.modes.exclusion(c1, c2);      else if (mode === 128) return p.modes.multiply(c1, c2);      else if (mode === 256) return p.modes.screen(c1, c2);      else if (mode === 1024) return p.modes.hard_light(c1, c2);      else if (mode === 2048) return p.modes.soft_light(c1, c2);      else if (mode === 512) return p.modes.overlay(c1, c2);      else if (mode === 4096) return p.modes.dodge(c1, c2);      else if (mode === 8192) return p.modes.burn(c1, c2)    };    function saveContext() {      curContext.save()    }    function restoreContext() {      curContext.restore();      isStrokeDirty = true;      isFillDirty = true    }    p.printMatrix = function() {      modelView.print()    };    Drawing2D.prototype.translate = function(x, y) {      modelView.translate(x, y);      modelViewInv.invTranslate(x, y);      curContext.translate(x, y)    };    Drawing3D.prototype.translate = function(x, y, z) {      modelView.translate(x, y, z);      modelViewInv.invTranslate(x, y, z)    };    Drawing2D.prototype.scale = function(x, y) {      modelView.scale(x, y);      modelViewInv.invScale(x, y);      curContext.scale(x, y || x)    };    Drawing3D.prototype.scale = function(x, y, z) {      modelView.scale(x, y, z);      modelViewInv.invScale(x, y, z)    };    Drawing2D.prototype.transform = function(pmatrix) {      var e = pmatrix.array();      curContext.transform(e[0], e[3], e[1], e[4], e[2], e[5])    };    Drawing3D.prototype.transformm = function(pmatrix3d) {      throw "p.transform is currently not supported in 3D mode";    };    Drawing2D.prototype.pushMatrix = function() {      userMatrixStack.load(modelView);      userReverseMatrixStack.load(modelViewInv);      saveContext()    };    Drawing3D.prototype.pushMatrix = function() {      userMatrixStack.load(modelView);      userReverseMatrixStack.load(modelViewInv)    };    Drawing2D.prototype.popMatrix = function() {      modelView.set(userMatrixStack.pop());      modelViewInv.set(userReverseMatrixStack.pop());      restoreContext()    };    Drawing3D.prototype.popMatrix = function() {      modelView.set(userMatrixStack.pop());      modelViewInv.set(userReverseMatrixStack.pop())    };    Drawing2D.prototype.resetMatrix = function() {      modelView.reset();      modelViewInv.reset();      curContext.setTransform(1, 0, 0, 1, 0, 0)    };    Drawing3D.prototype.resetMatrix = function() {      modelView.reset();      modelViewInv.reset()    };    DrawingShared.prototype.applyMatrix = function() {      var a = arguments;      modelView.apply(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]);      modelViewInv.invApply(a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15])    };    Drawing2D.prototype.applyMatrix = function() {      var a = arguments;      for (var cnt = a.length; cnt < 16; cnt++) a[cnt] = 0;      a[10] = a[15] = 1;      DrawingShared.prototype.applyMatrix.apply(this, a)    };    p.rotateX = function(angleInRadians) {      modelView.rotateX(angleInRadians);      modelViewInv.invRotateX(angleInRadians)    };    Drawing2D.prototype.rotateZ = function() {      throw "rotateZ() is not supported in 2D mode. Use rotate(float) instead.";    };    Drawing3D.prototype.rotateZ = function(angleInRadians) {      modelView.rotateZ(angleInRadians);      modelViewInv.invRotateZ(angleInRadians)    };    p.rotateY = function(angleInRadians) {      modelView.rotateY(angleInRadians);      modelViewInv.invRotateY(angleInRadians)    };    Drawing2D.prototype.rotate = function(angleInRadians) {      modelView.rotateZ(angleInRadians);      modelViewInv.invRotateZ(angleInRadians);      curContext.rotate(angleInRadians)    };    Drawing3D.prototype.rotate = function(angleInRadians) {      p.rotateZ(angleInRadians)    };    Drawing2D.prototype.shearX = function(angleInRadians) {      modelView.shearX(angleInRadians);      curContext.transform(1, 0, angleInRadians, 1, 0, 0)    };    Drawing3D.prototype.shearX = function(angleInRadians) {      modelView.shearX(angleInRadians)    };    Drawing2D.prototype.shearY = function(angleInRadians) {      modelView.shearY(angleInRadians);      curContext.transform(1, angleInRadians, 0, 1, 0, 0)    };    Drawing3D.prototype.shearY = function(angleInRadians) {      modelView.shearY(angleInRadians)    };    p.pushStyle = function() {      saveContext();      p.pushMatrix();      var newState = {        "doFill": doFill,        "currentFillColor": currentFillColor,        "doStroke": doStroke,        "currentStrokeColor": currentStrokeColor,        "curTint": curTint,        "curRectMode": curRectMode,        "curColorMode": curColorMode,        "colorModeX": colorModeX,        "colorModeZ": colorModeZ,        "colorModeY": colorModeY,        "colorModeA": colorModeA,        "curTextFont": curTextFont,        "horizontalTextAlignment": horizontalTextAlignment,        "verticalTextAlignment": verticalTextAlignment,        "textMode": textMode,        "curFontName": curFontName,        "curTextSize": curTextSize,        "curTextAscent": curTextAscent,        "curTextDescent": curTextDescent,        "curTextLeading": curTextLeading      };      styleArray.push(newState)    };    p.popStyle = function() {      var oldState = styleArray.pop();      if (oldState) {        restoreContext();        p.popMatrix();        doFill = oldState.doFill;        currentFillColor = oldState.currentFillColor;        doStroke = oldState.doStroke;        currentStrokeColor = oldState.currentStrokeColor;        curTint = oldState.curTint;        curRectMode = oldState.curRectMode;        curColorMode = oldState.curColorMode;        colorModeX = oldState.colorModeX;        colorModeZ = oldState.colorModeZ;        colorModeY = oldState.colorModeY;        colorModeA = oldState.colorModeA;        curTextFont = oldState.curTextFont;        curFontName = oldState.curFontName;        curTextSize = oldState.curTextSize;        horizontalTextAlignment = oldState.horizontalTextAlignment;        verticalTextAlignment = oldState.verticalTextAlignment;        textMode = oldState.textMode;        curTextAscent = oldState.curTextAscent;        curTextDescent = oldState.curTextDescent;        curTextLeading = oldState.curTextLeading      } else throw "Too many popStyle() without enough pushStyle()";    };    p.year = function() {      return (new Date).getFullYear()    };    p.month = function() {      return (new Date).getMonth() + 1    };    p.day = function() {      return (new Date).getDate()    };    p.hour = function() {      return (new Date).getHours()    };    p.minute = function() {      return (new Date).getMinutes()    };    p.second = function() {      return (new Date).getSeconds()    };    p.millis = function() {      return Date.now() - start    };    function redrawHelper() {      var sec = (Date.now() - timeSinceLastFPS) / 1E3;      framesSinceLastFPS++;      var fps = framesSinceLastFPS / sec;      if (sec > 0.5) {        timeSinceLastFPS = Date.now();        framesSinceLastFPS = 0;        p.__frameRate = fps      }      p.frameCount++    }    Drawing2D.prototype.redraw = function() {      redrawHelper();      curContext.lineWidth = lineWidth;      var pmouseXLastEvent = p.pmouseX,        pmouseYLastEvent = p.pmouseY;      p.pmouseX = pmouseXLastFrame;      p.pmouseY = pmouseYLastFrame;      saveContext();      p.draw();      restoreContext();      pmouseXLastFrame = p.mouseX;      pmouseYLastFrame = p.mouseY;      p.pmouseX = pmouseXLastEvent;      p.pmouseY = pmouseYLastEvent    };    Drawing3D.prototype.redraw = function() {      redrawHelper();      var pmouseXLastEvent = p.pmouseX,        pmouseYLastEvent = p.pmouseY;      p.pmouseX = pmouseXLastFrame;      p.pmouseY = pmouseYLastFrame;      curContext.clear(curContext.DEPTH_BUFFER_BIT);      curContextCache = {        attributes: {},        locations: {}      };      p.noLights();      p.lightFalloff(1, 0, 0);      p.shininess(1);      p.ambient(255, 255, 255);      p.specular(0, 0, 0);      p.emissive(0, 0, 0);      p.camera();      p.draw();      pmouseXLastFrame = p.mouseX;      pmouseYLastFrame = p.mouseY;      p.pmouseX = pmouseXLastEvent;      p.pmouseY = pmouseYLastEvent    };    p.noLoop = function() {      doLoop = false;      loopStarted = false;      clearInterval(looping);      curSketch.onPause()    };    p.loop = function() {      if (loopStarted) return;      timeSinceLastFPS = Date.now();      framesSinceLastFPS = 0;      looping = window.setInterval(function() {        try {          curSketch.onFrameStart();          p.redraw();          curSketch.onFrameEnd()        } catch(e_loop) {          window.clearInterval(looping);          throw e_loop;        }      },      curMsPerFrame);      doLoop = true;      loopStarted = true;      curSketch.onLoop()    };    p.frameRate = function(aRate) {      curFrameRate = aRate;      curMsPerFrame = 1E3 / curFrameRate;      if (doLoop) {        p.noLoop();        p.loop()      }    };    var eventHandlers = [];    function attachEventHandler(elem, type, fn) {      if (elem.addEventListener) elem.addEventListener(type, fn, false);      else elem.attachEvent("on" + type, fn);      eventHandlers.push({        elem: elem,        type: type,        fn: fn      })    }    function detachEventHandler(eventHandler) {      var elem = eventHandler.elem,        type = eventHandler.type,        fn = eventHandler.fn;      if (elem.removeEventListener) elem.removeEventListener(type, fn, false);      else if (elem.detachEvent) elem.detachEvent("on" + type, fn)    }    p.exit = function() {      window.clearInterval(looping);      removeInstance(p.externals.canvas.id);      delete curElement.onmousedown;      for (var lib in Processing.lib) if (Processing.lib.hasOwnProperty(lib)) if (Processing.lib[lib].hasOwnProperty("detach")) Processing.lib[lib].detach(p);      var i = eventHandlers.length;      while (i--) detachEventHandler(eventHandlers[i]);      curSketch.onExit()    };    p.cursor = function() {      if (arguments.length > 1 || arguments.length === 1 && arguments[0] instanceof p.PImage) {        var image = arguments[0],          x, y;        if (arguments.length >= 3) {          x = arguments[1];          y = arguments[2];          if (x < 0 || y < 0 || y >= image.height || x >= image.width) throw "x and y must be non-negative and less than the dimensions of the image";        } else {          x = image.width >>> 1;          y = image.height >>> 1        }        var imageDataURL = image.toDataURL();        var style = 'url("' + imageDataURL + '") ' + x + " " + y + ", default";        curCursor = curElement.style.cursor = style      } else if (arguments.length === 1) {        var mode = arguments[0];        curCursor = curElement.style.cursor = mode      } else curCursor = curElement.style.cursor = oldCursor    };    p.noCursor = function() {      curCursor = curElement.style.cursor = PConstants.NOCURSOR    };    p.link = function(href, target) {      if (target !== undef) window.open(href, target);      else window.location = href    };    p.beginDraw = nop;    p.endDraw = nop;    Drawing2D.prototype.toImageData = function(x, y, w, h) {      x = x !== undef ? x : 0;      y = y !== undef ? y : 0;      w = w !== undef ? w : p.width;      h = h !== undef ? h : p.height;      return curContext.getImageData(x, y, w, h)    };    Drawing3D.prototype.toImageData = function(x, y, w, h) {      x = x !== undef ? x : 0;      y = y !== undef ? y : 0;      w = w !== undef ? w : p.width;      h = h !== undef ? h : p.height;      var c = document.createElement("canvas"),        ctx = c.getContext("2d"),        obj = ctx.createImageData(w, h),        uBuff = new Uint8Array(w * h * 4);      curContext.readPixels(x, y, w, h, curContext.RGBA, curContext.UNSIGNED_BYTE, uBuff);      for (var i = 0, ul = uBuff.length, obj_data = obj.data; i < ul; i++) obj_data[i] = uBuff[(h - 1 - Math.floor(i / 4 / w)) * w * 4 + i % (w * 4)];      return obj    };    p.status = function(text) {      window.status = text    };    p.binary = function(num, numBits) {      var bit;      if (numBits > 0) bit = numBits;      else if (num instanceof Char) {        bit = 16;        num |= 0      } else {        bit = 32;        while (bit > 1 && !(num >>> bit - 1 & 1)) bit--      }      var result = "";      while (bit > 0) result += num >>> --bit & 1 ? "1" : "0";      return result    };    p.unbinary = function(binaryString) {      var i = binaryString.length - 1,        mask = 1,        result = 0;      while (i >= 0) {        var ch = binaryString[i--];        if (ch !== "0" && ch !== "1") throw "the value passed into unbinary was not an 8 bit binary number";        if (ch === "1") result += mask;        mask <<= 1      }      return result    };    function nfCoreScalar(value, plus, minus, leftDigits, rightDigits, group) {      var sign = value < 0 ? minus : plus;      var autoDetectDecimals = rightDigits === 0;      var rightDigitsOfDefault = rightDigits === undef || rightDigits < 0 ? 0 : rightDigits;      var absValue = Math.abs(value);      if (autoDetectDecimals) {        rightDigitsOfDefault = 1;        absValue *= 10;        while (Math.abs(Math.round(absValue) - absValue) > 1.0E-6 && rightDigitsOfDefault < 7) {          ++rightDigitsOfDefault;          absValue *= 10        }      } else if (rightDigitsOfDefault !== 0) absValue *= Math.pow(10, rightDigitsOfDefault);      var number, doubled = absValue * 2;      if (Math.floor(absValue) === absValue) number = absValue;      else if (Math.floor(doubled) === doubled) {        var floored = Math.floor(absValue);        number = floored + floored % 2      } else number = Math.round(absValue);      var buffer = "";      var totalDigits = leftDigits + rightDigitsOfDefault;      while (totalDigits > 0 || number > 0) {        totalDigits--;        buffer = "" + number % 10 + buffer;        number = Math.floor(number / 10)      }      if (group !== undef) {        var i = buffer.length - 3 - rightDigitsOfDefault;        while (i > 0) {          buffer = buffer.substring(0, i) + group + buffer.substring(i);          i -= 3        }      }      if (rightDigitsOfDefault > 0) return sign + buffer.substring(0, buffer.length - rightDigitsOfDefault) + "." + buffer.substring(buffer.length - rightDigitsOfDefault, buffer.length);      return sign + buffer    }    function nfCore(value, plus, minus, leftDigits, rightDigits, group) {      if (value instanceof Array) {        var arr = [];        for (var i = 0, len = value.length; i < len; i++) arr.push(nfCoreScalar(value[i], plus, minus, leftDigits, rightDigits, group));        return arr      }      return nfCoreScalar(value, plus, minus, leftDigits, rightDigits, group)    }    p.nf = function(value, leftDigits, rightDigits) {      return nfCore(value, "", "-", leftDigits, rightDigits)    };    p.nfs = function(value, leftDigits, rightDigits) {      return nfCore(value, " ", "-", leftDigits, rightDigits)    };    p.nfp = function(value, leftDigits, rightDigits) {      return nfCore(value, "+", "-", leftDigits, rightDigits)    };    p.nfc = function(value, leftDigits, rightDigits) {      return nfCore(value, "", "-", leftDigits, rightDigits, ",")    };    var decimalToHex = function(d, padding) {      padding = padding === undef || padding === null ? padding = 8 : padding;      if (d < 0) d = 4294967295 + d + 1;      var hex = Number(d).toString(16).toUpperCase();      while (hex.length < padding) hex = "0" + hex;      if (hex.length >= padding) hex = hex.substring(hex.length - padding, hex.length);      return hex    };    p.hex = function(value, len) {      if (arguments.length === 1) if (value instanceof Char) len = 4;      else len = 8;      return decimalToHex(value, len)    };    function unhexScalar(hex) {      var value = parseInt("0x" + hex, 16);      if (value > 2147483647) value -= 4294967296;      return value    }    p.unhex = function(hex) {      if (hex instanceof Array) {        var arr = [];        for (var i = 0; i < hex.length; i++) arr.push(unhexScalar(hex[i]));        return arr      }      return unhexScalar(hex)    };    p.loadStrings = function(filename) {      if (localStorage[filename]) return localStorage[filename].split("\n");      var filecontent = ajax(filename);      if (typeof filecontent !== "string" || filecontent === "") return [];      filecontent = filecontent.replace(/(\r\n?)/g, "\n").replace(/\n$/, "");      return filecontent.split("\n")    };    p.saveStrings = function(filename, strings) {      localStorage[filename] = strings.join("\n")    };    p.loadBytes = function(url) {      var string = ajax(url);      var ret = [];      for (var i = 0; i < string.length; i++) ret.push(string.charCodeAt(i));      return ret    };    function removeFirstArgument(args) {      return Array.prototype.slice.call(args, 1)    }    p.matchAll = function(aString, aRegExp) {      var results = [],        latest;      var regexp = new RegExp(aRegExp, "g");      while ((latest = regexp.exec(aString)) !== null) {        results.push(latest);        if (latest[0].length === 0)++regexp.lastIndex      }      return results.length > 0 ? results : null    };    p.__contains = function(subject, subStr) {      if (typeof subject !== "string") return subject.contains.apply(subject, removeFirstArgument(arguments));      return subject !== null && subStr !== null && typeof subStr === "string" && subject.indexOf(subStr) > -1    };    p.__replaceAll = function(subject, regex, replacement) {      if (typeof subject !== "string") return subject.replaceAll.apply(subject, removeFirstArgument(arguments));      return subject.replace(new RegExp(regex, "g"), replacement)    };    p.__replaceFirst = function(subject, regex, replacement) {      if (typeof subject !== "string") return subject.replaceFirst.apply(subject, removeFirstArgument(arguments));      return subject.replace(new RegExp(regex, ""), replacement)    };    p.__replace = function(subject, what, replacement) {      if (typeof subject !== "string") return subject.replace.apply(subject, removeFirstArgument(arguments));      if (what instanceof RegExp) return subject.replace(what, replacement);      if (typeof what !== "string") what = what.toString();      if (what === "") return subject;      var i = subject.indexOf(what);      if (i < 0) return subject;      var j = 0,        result = "";      do {        result += subject.substring(j, i) + replacement;        j = i + what.length      } while ((i = subject.indexOf(what, j)) >= 0);      return result + subject.substring(j)    };    p.__equals = function(subject, other) {      if (subject.equals instanceof      Function) return subject.equals.apply(subject, removeFirstArgument(arguments));      return subject.valueOf() === other.valueOf()    };    p.__equalsIgnoreCase = function(subject, other) {      if (typeof subject !== "string") return subject.equalsIgnoreCase.apply(subject, removeFirstArgument(arguments));      return subject.toLowerCase() === other.toLowerCase()    };    p.__toCharArray = function(subject) {      if (typeof subject !== "string") return subject.toCharArray.apply(subject, removeFirstArgument(arguments));      var chars = [];      for (var i = 0, len = subject.length; i < len; ++i) chars[i] = new Char(subject.charAt(i));      return chars    };    p.__split = function(subject, regex, limit) {      if (typeof subject !== "string") return subject.split.apply(subject, removeFirstArgument(arguments));      var pattern = new RegExp(regex);      if (limit === undef || limit < 1) return subject.split(pattern);      var result = [],        currSubject = subject,        pos;      while ((pos = currSubject.search(pattern)) !== -1 && result.length < limit - 1) {        var match = pattern.exec(currSubject).toString();        result.push(currSubject.substring(0, pos));        currSubject = currSubject.substring(pos + match.length)      }      if (pos !== -1 || currSubject !== "") result.push(currSubject);      return result    };    p.__codePointAt = function(subject, idx) {      var code = subject.charCodeAt(idx),        hi, low;      if (55296 <= code && code <= 56319) {        hi = code;        low = subject.charCodeAt(idx + 1);        return (hi - 55296) * 1024 + (low - 56320) + 65536      }      return code    };    p.match = function(str, regexp) {      return str.match(regexp)    };    p.__matches = function(str, regexp) {      return (new RegExp(regexp)).test(str)    };    p.__startsWith = function(subject, prefix, toffset) {      if (typeof subject !== "string") return subject.startsWith.apply(subject, removeFirstArgument(arguments));      toffset = toffset || 0;      if (toffset < 0 || toffset > subject.length) return false;      return prefix === "" || prefix === subject ? true : subject.indexOf(prefix) === toffset    };    p.__endsWith = function(subject, suffix) {      if (typeof subject !== "string") return subject.endsWith.apply(subject, removeFirstArgument(arguments));      var suffixLen = suffix ? suffix.length : 0;      return suffix === "" || suffix === subject ? true : subject.indexOf(suffix) === subject.length - suffixLen    };    p.__hashCode = function(subject) {      if (subject.hashCode instanceof      Function) return subject.hashCode.apply(subject, removeFirstArgument(arguments));      return virtHashCode(subject)    };    p.__printStackTrace = function(subject) {      p.println("Exception: " + subject.toString())    };    var logBuffer = [];    p.println = function(message) {      var bufferLen = logBuffer.length;      if (bufferLen) {        Processing.logger.log(logBuffer.join(""));        logBuffer.length = 0      }      if (arguments.length === 0 && bufferLen === 0) Processing.logger.log("");      else if (arguments.length !== 0) Processing.logger.log(message)    };    p.print = function(message) {      logBuffer.push(message)    };    p.str = function(val) {      if (val instanceof Array) {        var arr = [];        for (var i = 0; i < val.length; i++) arr.push(val[i].toString() + "");        return arr      }      return val.toString() + ""    };    p.trim = function(str) {      if (str instanceof Array) {        var arr = [];        for (var i = 0; i < str.length; i++) arr.push(str[i].replace(/^\s*/, "").replace(/\s*$/, "").replace(/\r*$/, ""));        return arr      }      return str.replace(/^\s*/, "").replace(/\s*$/, "").replace(/\r*$/, "")    };    function booleanScalar(val) {      if (typeof val === "number") return val !== 0;      if (typeof val === "boolean") return val;      if (typeof val === "string") return val.toLowerCase() === "true";      if (val instanceof Char) return val.code === 49 || val.code === 84 || val.code === 116    }    p.parseBoolean = function(val) {      if (val instanceof Array) {        var ret = [];        for (var i = 0; i < val.length; i++) ret.push(booleanScalar(val[i]));        return ret      }      return booleanScalar(val)    };    p.parseByte = function(what) {      if (what instanceof Array) {        var bytes = [];        for (var i = 0; i < what.length; i++) bytes.push(0 - (what[i] & 128) | what[i] & 127);        return bytes      }      return 0 - (what & 128) | what & 127    };    p.parseChar = function(key) {      if (typeof key === "number") return new Char(String.fromCharCode(key & 65535));      if (key instanceof Array) {        var ret = [];        for (var i = 0; i < key.length; i++) ret.push(new Char(String.fromCharCode(key[i] & 65535)));        return ret      }      throw "char() may receive only one argument of type int, byte, int[], or byte[].";    };    function floatScalar(val) {      if (typeof val === "number") return val;      if (typeof val === "boolean") return val ? 1 : 0;      if (typeof val === "string") return parseFloat(val);      if (val instanceof Char) return val.code    }    p.parseFloat = function(val) {      if (val instanceof      Array) {        var ret = [];        for (var i = 0; i < val.length; i++) ret.push(floatScalar(val[i]));        return ret      }      return floatScalar(val)    };    function intScalar(val, radix) {      if (typeof val === "number") return val & 4294967295;      if (typeof val === "boolean") return val ? 1 : 0;      if (typeof val === "string") {        var number = parseInt(val, radix || 10);        return number & 4294967295      }      if (val instanceof Char) return val.code    }    p.parseInt = function(val, radix) {      if (val instanceof Array) {        var ret = [];        for (var i = 0; i < val.length; i++) if (typeof val[i] === "string" && !/^\s*[+\-]?\d+\s*$/.test(val[i])) ret.push(0);        else ret.push(intScalar(val[i], radix));        return ret      }      return intScalar(val, radix)    };    p.__int_cast = function(val) {      return 0 | val    };    p.__instanceof = function(obj, type) {      if (typeof type !== "function") throw "Function is expected as type argument for instanceof operator";      if (typeof obj === "string") return type === Object || type === String;      if (obj instanceof type) return true;      if (typeof obj !== "object" || obj === null) return false;      var objType = obj.constructor;      if (type.$isInterface) {        var interfaces = [];        while (objType) {          if (objType.$interfaces) interfaces = interfaces.concat(objType.$interfaces);          objType = objType.$base        }        while (interfaces.length > 0) {          var i = interfaces.shift();          if (i === type) return true;          if (i.$interfaces) interfaces = interfaces.concat(i.$interfaces)        }        return false      }      while (objType.hasOwnProperty("$base")) {        objType = objType.$base;        if (objType === type) return true      }      return false    };    p.abs = Math.abs;    p.ceil = Math.ceil;    p.constrain = function(aNumber, aMin, aMax) {      return aNumber > aMax ? aMax : aNumber < aMin ? aMin : aNumber    };    p.dist = function() {      var dx, dy, dz;      if (arguments.length === 4) {        dx = arguments[0] - arguments[2];        dy = arguments[1] - arguments[3];        return Math.sqrt(dx * dx + dy * dy)      }      if (arguments.length === 6) {        dx = arguments[0] - arguments[3];        dy = arguments[1] - arguments[4];        dz = arguments[2] - arguments[5];        return Math.sqrt(dx * dx + dy * dy + dz * dz)      }    };    p.exp = Math.exp;    p.floor = Math.floor;    p.lerp = function(value1, value2, amt) {      return (value2 - value1) * amt + value1    };    p.log = Math.log;    p.mag = function(a, b, c) {      if (c) return Math.sqrt(a * a + b * b + c * c);      return Math.sqrt(a * a + b * b)    };    p.map = function(value, istart, istop, ostart, ostop) {      return ostart + (ostop - ostart) * ((value - istart) / (istop - istart))    };    p.max = function() {      if (arguments.length === 2) return arguments[0] < arguments[1] ? arguments[1] : arguments[0];      var numbers = arguments.length === 1 ? arguments[0] : arguments;      if (! ("length" in numbers && numbers.length > 0)) throw "Non-empty array is expected";      var max = numbers[0],        count = numbers.length;      for (var i = 1; i < count; ++i) if (max < numbers[i]) max = numbers[i];      return max    };    p.min = function() {      if (arguments.length === 2) return arguments[0] < arguments[1] ? arguments[0] : arguments[1];      var numbers = arguments.length === 1 ? arguments[0] : arguments;      if (! ("length" in numbers && numbers.length > 0)) throw "Non-empty array is expected";      var min = numbers[0],        count = numbers.length;      for (var i = 1; i < count; ++i) if (min > numbers[i]) min = numbers[i];      return min    };    p.norm = function(aNumber, low, high) {      return (aNumber - low) / (high - low)    };    p.pow = Math.pow;    p.round = Math.round;    p.sq = function(aNumber) {      return aNumber * aNumber    };    p.sqrt = Math.sqrt;    p.acos = Math.acos;    p.asin = Math.asin;    p.atan = Math.atan;    p.atan2 = Math.atan2;    p.cos = Math.cos;    p.degrees = function(aAngle) {      return aAngle * 180 / Math.PI    };    p.radians = function(aAngle) {      return aAngle / 180 * Math.PI    };    p.sin = Math.sin;    p.tan = Math.tan;    var currentRandom = Math.random;    p.random = function() {      if (arguments.length === 0) return currentRandom();      if (arguments.length === 1) return currentRandom() * arguments[0];      var aMin = arguments[0],        aMax = arguments[1];      return currentRandom() * (aMax - aMin) + aMin    };    function Marsaglia(i1, i2) {      var z = i1 || 362436069,        w = i2 || 521288629;      var nextInt = function() {        z = 36969 * (z & 65535) + (z >>> 16) & 4294967295;        w = 18E3 * (w & 65535) + (w >>> 16) & 4294967295;        return ((z & 65535) << 16 | w & 65535) & 4294967295      };      this.nextDouble = function() {        var i = nextInt() / 4294967296;        return i < 0 ? 1 + i : i      };      this.nextInt = nextInt    }    Marsaglia.createRandomized = function() {      var now = new Date;      return new Marsaglia(now / 6E4 & 4294967295, now & 4294967295)    };    p.randomSeed = function(seed) {      currentRandom = (new Marsaglia(seed)).nextDouble    };    p.Random = function(seed) {      var haveNextNextGaussian = false,        nextNextGaussian, random;      this.nextGaussian = function() {        if (haveNextNextGaussian) {          haveNextNextGaussian = false;          return nextNextGaussian        }        var v1, v2, s;        do {          v1 = 2 * random() - 1;          v2 = 2 * random() - 1;          s = v1 * v1 + v2 * v2        } while (s >= 1 || s === 0);        var multiplier = Math.sqrt(-2 * Math.log(s) / s);        nextNextGaussian = v2 * multiplier;        haveNextNextGaussian = true;        return v1 * multiplier      };      random = seed === undef ? Math.random : (new Marsaglia(seed)).nextDouble    };    function PerlinNoise(seed) {      var rnd = seed !== undef ? new Marsaglia(seed) : Marsaglia.createRandomized();      var i, j;      var perm = new Uint8Array(512);      for (i = 0; i < 256; ++i) perm[i] = i;      for (i = 0; i < 256; ++i) {        var t = perm[j = rnd.nextInt() & 255];        perm[j] = perm[i];        perm[i] = t      }      for (i = 0; i < 256; ++i) perm[i + 256] = perm[i];      function grad3d(i, x, y, z) {        var h = i & 15;        var u = h < 8 ? x : y,        v = h < 4 ? y : h === 12 || h === 14 ? x : z;        return ((h & 1) === 0 ? u : -u) + ((h & 2) === 0 ? v : -v)      }      function grad2d(i, x, y) {        var v = (i & 1) === 0 ? x : y;        return (i & 2) === 0 ? -v : v      }      function grad1d(i, x) {        return (i & 1) === 0 ? -x : x      }      function lerp(t, a, b) {        return a + t * (b - a)      }      this.noise3d = function(x, y, z) {        var X = Math.floor(x) & 255,          Y = Math.floor(y) & 255,          Z = Math.floor(z) & 255;        x -= Math.floor(x);        y -= Math.floor(y);        z -= Math.floor(z);        var fx = (3 - 2 * x) * x * x,          fy = (3 - 2 * y) * y * y,          fz = (3 - 2 * z) * z * z;        var p0 = perm[X] + Y,          p00 = perm[p0] + Z,          p01 = perm[p0 + 1] + Z,          p1 = perm[X + 1] + Y,          p10 = perm[p1] + Z,          p11 = perm[p1 + 1] + Z;        return lerp(fz, lerp(fy, lerp(fx, grad3d(perm[p00], x, y, z), grad3d(perm[p10], x - 1, y, z)), lerp(fx, grad3d(perm[p01], x, y - 1, z), grad3d(perm[p11], x - 1, y - 1, z))), lerp(fy, lerp(fx, grad3d(perm[p00 + 1], x, y, z - 1), grad3d(perm[p10 + 1], x - 1, y, z - 1)), lerp(fx, grad3d(perm[p01 + 1], x, y - 1, z - 1), grad3d(perm[p11 + 1], x - 1, y - 1, z - 1))))      };      this.noise2d = function(x, y) {        var X = Math.floor(x) & 255,          Y = Math.floor(y) & 255;        x -= Math.floor(x);        y -= Math.floor(y);        var fx = (3 - 2 * x) * x * x,          fy = (3 - 2 * y) * y * y;        var p0 = perm[X] + Y,          p1 = perm[X + 1] + Y;        return lerp(fy, lerp(fx, grad2d(perm[p0], x, y), grad2d(perm[p1], x - 1, y)), lerp(fx, grad2d(perm[p0 + 1], x, y - 1), grad2d(perm[p1 + 1], x - 1, y - 1)))      };      this.noise1d = function(x) {        var X = Math.floor(x) & 255;        x -= Math.floor(x);        var fx = (3 - 2 * x) * x * x;        return lerp(fx, grad1d(perm[X], x), grad1d(perm[X + 1], x - 1))      }    }    var noiseProfile = {      generator: undef,      octaves: 4,      fallout: 0.5,      seed: undef    };    p.noise = function(x, y, z) {      if (noiseProfile.generator === undef) noiseProfile.generator = new PerlinNoise(noiseProfile.seed);      var generator = noiseProfile.generator;      var effect = 1,        k = 1,        sum = 0;      for (var i = 0; i < noiseProfile.octaves; ++i) {        effect *= noiseProfile.fallout;        switch (arguments.length) {        case 1:          sum += effect * (1 + generator.noise1d(k * x)) / 2;          break;        case 2:          sum += effect * (1 + generator.noise2d(k * x, k * y)) / 2;          break;        case 3:          sum += effect * (1 + generator.noise3d(k * x, k * y, k * z)) / 2;          break        }        k *= 2      }      return sum    };    p.noiseDetail = function(octaves, fallout) {      noiseProfile.octaves = octaves;      if (fallout !== undef) noiseProfile.fallout = fallout    };    p.noiseSeed = function(seed) {      noiseProfile.seed = seed;      noiseProfile.generator = undef    };    DrawingShared.prototype.size = function(aWidth, aHeight, aMode) {      if (doStroke) p.stroke(0);      if (doFill) p.fill(255);      var savedProperties = {        fillStyle: curContext.fillStyle,        strokeStyle: curContext.strokeStyle,        lineCap: curContext.lineCap,        lineJoin: curContext.lineJoin      };      if (curElement.style.length > 0) {        curElement.style.removeProperty("width");        curElement.style.removeProperty("height")      }      curElement.width = p.width = aWidth || 100;      curElement.height = p.height = aHeight || 100;      for (var prop in savedProperties) if (savedProperties.hasOwnProperty(prop)) curContext[prop] = savedProperties[prop];      p.textFont(curTextFont);      p.background();      maxPixelsCached = Math.max(1E3, aWidth * aHeight * 0.05);      p.externals.context = curContext;      for (var i = 0; i < 720; i++) {        sinLUT[i] = p.sin(i * (Math.PI / 180) * 0.5);        cosLUT[i] = p.cos(i * (Math.PI / 180) * 0.5)      }    };    Drawing2D.prototype.size = function(aWidth, aHeight, aMode) {      if (curContext === undef) {        curContext = curElement.getContext("2d");        userMatrixStack = new PMatrixStack;        userReverseMatrixStack = new PMatrixStack;        modelView = new PMatrix2D;        modelViewInv = new PMatrix2D      }      DrawingShared.prototype.size.apply(this, arguments)    };    Drawing3D.prototype.size = function() {      var size3DCalled = false;      return function size(aWidth, aHeight, aMode) {        if (size3DCalled) throw "Multiple calls to size() for 3D renders are not allowed.";        size3DCalled = true;        function getGLContext(canvas) {          var ctxNames = ["experimental-webgl", "webgl", "webkit-3d"],            gl;          for (var i = 0, l = ctxNames.length; i < l; i++) {            gl = canvas.getContext(ctxNames[i], {              antialias: false,              preserveDrawingBuffer: true            });            if (gl) break          }          return gl        }        try {          curElement.width = p.width = aWidth || 100;          curElement.height = p.height = aHeight || 100;          curContext = getGLContext(curElement);          canTex = curContext.createTexture();          textTex = curContext.createTexture()        } catch(e_size) {          Processing.debug(e_size)        }        if (!curContext) throw "WebGL context is not supported on this browser.";        curContext.viewport(0, 0, curElement.width, curElement.height);        curContext.enable(curContext.DEPTH_TEST);        curContext.enable(curContext.BLEND);        curContext.blendFunc(curContext.SRC_ALPHA, curContext.ONE_MINUS_SRC_ALPHA);        programObject2D = createProgramObject(curContext, vertexShaderSrc2D, fragmentShaderSrc2D);        programObjectUnlitShape = createProgramObject(curContext, vertexShaderSrcUnlitShape, fragmentShaderSrcUnlitShape);        p.strokeWeight(1);        programObject3D = createProgramObject(curContext, vertexShaderSrc3D, fragmentShaderSrc3D);        curContext.useProgram(programObject3D);        uniformi("usingTexture3d", programObject3D, "usingTexture", usingTexture);        p.lightFalloff(1, 0, 0);        p.shininess(1);        p.ambient(255, 255, 255);        p.specular(0, 0, 0);        p.emissive(0, 0, 0);        boxBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, boxBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, boxVerts, curContext.STATIC_DRAW);        boxNormBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, boxNormBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, boxNorms, curContext.STATIC_DRAW);        boxOutlineBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, boxOutlineBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, boxOutlineVerts, curContext.STATIC_DRAW);        rectBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, rectBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, rectVerts, curContext.STATIC_DRAW);        rectNormBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, rectNormBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, rectNorms, curContext.STATIC_DRAW);        sphereBuffer = curContext.createBuffer();        lineBuffer = curContext.createBuffer();        fillBuffer = curContext.createBuffer();        fillColorBuffer = curContext.createBuffer();        strokeColorBuffer = curContext.createBuffer();        shapeTexVBO = curContext.createBuffer();        pointBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, pointBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array([0, 0, 0]), curContext.STATIC_DRAW);        textBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, textBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array([1, 1, 0, -1, 1, 0, -1, -1, 0, 1, -1, 0]), curContext.STATIC_DRAW);        textureBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ARRAY_BUFFER, textureBuffer);        curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array([0, 0, 1, 0, 1, 1, 0, 1]), curContext.STATIC_DRAW);        indexBuffer = curContext.createBuffer();        curContext.bindBuffer(curContext.ELEMENT_ARRAY_BUFFER, indexBuffer);        curContext.bufferData(curContext.ELEMENT_ARRAY_BUFFER, new Uint16Array([0, 1, 2, 2, 3, 0]), curContext.STATIC_DRAW);        cam = new PMatrix3D;        cameraInv = new PMatrix3D;        modelView = new PMatrix3D;        modelViewInv = new PMatrix3D;        projection = new PMatrix3D;        p.camera();        p.perspective();        userMatrixStack = new PMatrixStack;        userReverseMatrixStack = new PMatrixStack;        curveBasisMatrix = new PMatrix3D;        curveToBezierMatrix = new PMatrix3D;        curveDrawMatrix = new PMatrix3D;        bezierDrawMatrix = new PMatrix3D;        bezierBasisInverse = new PMatrix3D;        bezierBasisMatrix = new PMatrix3D;        bezierBasisMatrix.set(-1, 3, -3, 1, 3, -6, 3, 0, -3, 3, 0, 0, 1, 0, 0, 0);        DrawingShared.prototype.size.apply(this, arguments)      }    }();    Drawing2D.prototype.ambientLight = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.ambientLight = function(r, g, b, x, y, z) {      if (lightCount === 8) throw "can only create " + 8 + " lights";      var pos = new PVector(x, y, z);      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.mult(pos, pos);      var col = color$4(r, g, b, 0);      var normalizedCol = [((col >> 16) & 255) / 255, ((col >> 8) & 255) / 255, (col & 255) / 255];      curContext.useProgram(programObject3D);      uniformf("uLights.color.3d." + lightCount, programObject3D, "uLights" + lightCount + ".color", normalizedCol);      uniformf("uLights.position.3d." + lightCount, programObject3D, "uLights" + lightCount + ".position", pos.array());      uniformi("uLights.type.3d." + lightCount, programObject3D, "uLights" + lightCount + ".type", 0);      uniformi("uLightCount3d", programObject3D, "uLightCount", ++lightCount)    };    Drawing2D.prototype.directionalLight = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.directionalLight = function(r, g, b, nx, ny, nz) {      if (lightCount === 8) throw "can only create " + 8 + " lights";      curContext.useProgram(programObject3D);      var mvm = new PMatrix3D;      mvm.scale(1, -1, 1);      mvm.apply(modelView.array());      mvm = mvm.array();      var dir = [mvm[0] * nx + mvm[4] * ny + mvm[8] * nz, mvm[1] * nx + mvm[5] * ny + mvm[9] * nz, mvm[2] * nx + mvm[6] * ny + mvm[10] * nz];      var col = color$4(r, g, b, 0);      var normalizedCol = [((col >> 16) & 255) / 255, ((col >> 8) & 255) / 255, (col & 255) / 255];      uniformf("uLights.color.3d." + lightCount, programObject3D, "uLights" + lightCount + ".color", normalizedCol);      uniformf("uLights.position.3d." + lightCount, programObject3D, "uLights" + lightCount + ".position", dir);      uniformi("uLights.type.3d." + lightCount, programObject3D, "uLights" + lightCount + ".type", 1);      uniformi("uLightCount3d", programObject3D, "uLightCount", ++lightCount)    };    Drawing2D.prototype.lightFalloff = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.lightFalloff = function(constant, linear, quadratic) {      curContext.useProgram(programObject3D);      uniformf("uFalloff3d", programObject3D, "uFalloff", [constant, linear, quadratic])    };    Drawing2D.prototype.lightSpecular = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.lightSpecular = function(r, g, b) {      var col = color$4(r, g, b, 0);      var normalizedCol = [((col >> 16) & 255) / 255, ((col >> 8) & 255) / 255, (col & 255) / 255];      curContext.useProgram(programObject3D);      uniformf("uSpecular3d", programObject3D, "uSpecular", normalizedCol)    };    p.lights = function() {      p.ambientLight(128, 128, 128);      p.directionalLight(128, 128, 128, 0, 0, -1);      p.lightFalloff(1, 0, 0);      p.lightSpecular(0, 0, 0)    };    Drawing2D.prototype.pointLight = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.pointLight = function(r, g, b, x, y, z) {      if (lightCount === 8) throw "can only create " + 8 + " lights";      var pos = new PVector(x, y, z);      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.mult(pos, pos);      var col = color$4(r, g, b, 0);      var normalizedCol = [((col >> 16) & 255) / 255, ((col >> 8) & 255) / 255, (col & 255) / 255];      curContext.useProgram(programObject3D);      uniformf("uLights.color.3d." + lightCount, programObject3D, "uLights" + lightCount + ".color", normalizedCol);      uniformf("uLights.position.3d." + lightCount, programObject3D, "uLights" + lightCount + ".position", pos.array());      uniformi("uLights.type.3d." + lightCount, programObject3D, "uLights" + lightCount + ".type", 2);      uniformi("uLightCount3d", programObject3D, "uLightCount", ++lightCount)    };    Drawing2D.prototype.noLights = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.noLights = function() {      lightCount = 0;      curContext.useProgram(programObject3D);      uniformi("uLightCount3d", programObject3D, "uLightCount", lightCount)    };    Drawing2D.prototype.spotLight = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.spotLight = function(r, g, b, x, y, z, nx, ny, nz, angle, concentration) {      if (lightCount === 8) throw "can only create " + 8 + " lights";      curContext.useProgram(programObject3D);      var pos = new PVector(x, y, z);      var mvm = new PMatrix3D;      mvm.scale(1, -1, 1);      mvm.apply(modelView.array());      mvm.mult(pos, pos);      mvm = mvm.array();      var dir = [mvm[0] * nx + mvm[4] * ny + mvm[8] * nz, mvm[1] *        nx + mvm[5] * ny + mvm[9] * nz, mvm[2] * nx + mvm[6] * ny + mvm[10] * nz];      var col = color$4(r, g, b, 0);      var normalizedCol = [((col >> 16) & 255) / 255, ((col >> 8) & 255) / 255, (col & 255) / 255];      uniformf("uLights.color.3d." + lightCount, programObject3D, "uLights" + lightCount + ".color", normalizedCol);      uniformf("uLights.position.3d." + lightCount, programObject3D, "uLights" + lightCount + ".position", pos.array());      uniformf("uLights.direction.3d." + lightCount, programObject3D, "uLights" + lightCount + ".direction", dir);      uniformf("uLights.concentration.3d." + lightCount, programObject3D, "uLights" + lightCount + ".concentration", concentration);      uniformf("uLights.angle.3d." + lightCount, programObject3D, "uLights" + lightCount + ".angle", angle);      uniformi("uLights.type.3d." + lightCount, programObject3D, "uLights" + lightCount + ".type", 3);      uniformi("uLightCount3d", programObject3D, "uLightCount", ++lightCount)    };    Drawing2D.prototype.beginCamera = function() {      throw "beginCamera() is not available in 2D mode";    };    Drawing3D.prototype.beginCamera = function() {      if (manipulatingCamera) throw "You cannot call beginCamera() again before calling endCamera()";      manipulatingCamera = true;      modelView = cameraInv;      modelViewInv = cam    };    Drawing2D.prototype.endCamera = function() {      throw "endCamera() is not available in 2D mode";    };    Drawing3D.prototype.endCamera = function() {      if (!manipulatingCamera) throw "You cannot call endCamera() before calling beginCamera()";      modelView.set(cam);      modelViewInv.set(cameraInv);      manipulatingCamera = false    };    p.camera = function(eyeX, eyeY, eyeZ, centerX, centerY, centerZ, upX, upY, upZ) {      if (eyeX === undef) {        cameraX = p.width / 2;        cameraY = p.height / 2;        cameraZ = cameraY / Math.tan(cameraFOV / 2);        eyeX = cameraX;        eyeY = cameraY;        eyeZ = cameraZ;        centerX = cameraX;        centerY = cameraY;        centerZ = 0;        upX = 0;        upY = 1;        upZ = 0      }      var z = new PVector(eyeX - centerX, eyeY - centerY, eyeZ - centerZ);      var y = new PVector(upX, upY, upZ);      z.normalize();      var x = PVector.cross(y, z);      y = PVector.cross(z, x);      x.normalize();      y.normalize();      var xX = x.x,        xY = x.y,        xZ = x.z;      var yX = y.x,        yY = y.y,        yZ = y.z;      var zX = z.x,        zY = z.y,        zZ = z.z;      cam.set(xX, xY, xZ, 0, yX, yY, yZ, 0, zX, zY, zZ, 0, 0, 0, 0, 1);      cam.translate(-eyeX, -eyeY, -eyeZ);      cameraInv.reset();      cameraInv.invApply(xX, xY, xZ, 0, yX, yY, yZ, 0, zX, zY, zZ, 0, 0, 0, 0, 1);      cameraInv.translate(eyeX, eyeY, eyeZ);      modelView.set(cam);      modelViewInv.set(cameraInv)    };    p.perspective = function(fov, aspect, near, far) {      if (arguments.length === 0) {        cameraY = curElement.height / 2;        cameraZ = cameraY / Math.tan(cameraFOV / 2);        cameraNear = cameraZ / 10;        cameraFar = cameraZ * 10;        cameraAspect = p.width / p.height;        fov = cameraFOV;        aspect = cameraAspect;        near = cameraNear;        far = cameraFar      }      var yMax, yMin, xMax, xMin;      yMax = near * Math.tan(fov / 2);      yMin = -yMax;      xMax = yMax * aspect;      xMin = yMin * aspect;      p.frustum(xMin, xMax, yMin, yMax, near, far)    };    Drawing2D.prototype.frustum = function() {      throw "Processing.js: frustum() is not supported in 2D mode";    };    Drawing3D.prototype.frustum = function(left, right, bottom, top, near, far) {      frustumMode = true;      projection = new PMatrix3D;      projection.set(2 * near / (right - left), 0, (right + left) / (right - left), 0, 0, 2 * near / (top - bottom), (top + bottom) / (top - bottom), 0, 0, 0, -(far + near) / (far - near), -(2 * far * near) / (far - near), 0, 0, -1, 0);      var proj = new PMatrix3D;      proj.set(projection);      proj.transpose();      curContext.useProgram(programObject2D);      uniformMatrix("projection2d", programObject2D, "uProjection", false, proj.array());      curContext.useProgram(programObject3D);      uniformMatrix("projection3d", programObject3D, "uProjection", false, proj.array());      curContext.useProgram(programObjectUnlitShape);      uniformMatrix("uProjectionUS", programObjectUnlitShape, "uProjection", false, proj.array())    };    p.ortho = function(left, right, bottom, top, near, far) {      if (arguments.length === 0) {        left = 0;        right = p.width;        bottom = 0;        top = p.height;        near = -10;        far = 10      }      var x = 2 / (right - left);      var y = 2 / (top - bottom);      var z = -2 / (far - near);      var tx = -(right + left) / (right - left);      var ty = -(top + bottom) / (top - bottom);      var tz = -(far + near) / (far - near);      projection = new PMatrix3D;      projection.set(x, 0, 0, tx, 0, y, 0, ty, 0, 0, z, tz, 0, 0, 0, 1);      var proj = new PMatrix3D;      proj.set(projection);      proj.transpose();      curContext.useProgram(programObject2D);      uniformMatrix("projection2d", programObject2D, "uProjection", false, proj.array());      curContext.useProgram(programObject3D);      uniformMatrix("projection3d", programObject3D, "uProjection", false, proj.array());      curContext.useProgram(programObjectUnlitShape);      uniformMatrix("uProjectionUS", programObjectUnlitShape, "uProjection", false, proj.array());      frustumMode = false    };    p.printProjection = function() {      projection.print()    };    p.printCamera = function() {      cam.print()    };    Drawing2D.prototype.box = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.box = function(w, h, d) {      if (!h || !d) h = d = w;      var model = new PMatrix3D;      model.scale(w, h, d);      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      if (doFill) {        curContext.useProgram(programObject3D);        uniformMatrix("model3d", programObject3D, "uModel", false, model.array());        uniformMatrix("view3d", programObject3D, "uView", false, view.array());        curContext.enable(curContext.POLYGON_OFFSET_FILL);        curContext.polygonOffset(1, 1);        uniformf("color3d", programObject3D, "uColor", fillStyle);        if (lightCount > 0) {          var v = new PMatrix3D;          v.set(view);          var m = new PMatrix3D;          m.set(model);          v.mult(m);          var normalMatrix = new PMatrix3D;          normalMatrix.set(v);          normalMatrix.invert();          normalMatrix.transpose();          uniformMatrix("uNormalTransform3d", programObject3D, "uNormalTransform", false, normalMatrix.array());          vertexAttribPointer("aNormal3d", programObject3D, "aNormal", 3, boxNormBuffer)        } else disableVertexAttribPointer("aNormal3d", programObject3D, "aNormal");        vertexAttribPointer("aVertex3d", programObject3D, "aVertex", 3, boxBuffer);        disableVertexAttribPointer("aColor3d", programObject3D, "aColor");        disableVertexAttribPointer("aTexture3d", programObject3D, "aTexture");        curContext.drawArrays(curContext.TRIANGLES, 0, boxVerts.length / 3);        curContext.disable(curContext.POLYGON_OFFSET_FILL)      }      if (lineWidth > 0 && doStroke) {        curContext.useProgram(programObject2D);        uniformMatrix("uModel2d", programObject2D, "uModel", false, model.array());        uniformMatrix("uView2d", programObject2D, "uView", false, view.array());        uniformf("uColor2d", programObject2D, "uColor", strokeStyle);        uniformi("uIsDrawingText2d", programObject2D, "uIsDrawingText", false);        vertexAttribPointer("vertex2d", programObject2D, "aVertex", 3, boxOutlineBuffer);        disableVertexAttribPointer("aTextureCoord2d", programObject2D, "aTextureCoord");        curContext.drawArrays(curContext.LINES, 0, boxOutlineVerts.length / 3)      }    };    var initSphere = function() {      var i;      sphereVerts = [];      for (i = 0; i < sphereDetailU; i++) {        sphereVerts.push(0);        sphereVerts.push(-1);        sphereVerts.push(0);        sphereVerts.push(sphereX[i]);        sphereVerts.push(sphereY[i]);        sphereVerts.push(sphereZ[i])      }      sphereVerts.push(0);      sphereVerts.push(-1);      sphereVerts.push(0);      sphereVerts.push(sphereX[0]);      sphereVerts.push(sphereY[0]);      sphereVerts.push(sphereZ[0]);      var v1, v11, v2;      var voff = 0;      for (i = 2; i < sphereDetailV; i++) {        v1 = v11 = voff;        voff += sphereDetailU;        v2 = voff;        for (var j = 0; j < sphereDetailU; j++) {          sphereVerts.push(sphereX[v1]);          sphereVerts.push(sphereY[v1]);          sphereVerts.push(sphereZ[v1++]);          sphereVerts.push(sphereX[v2]);          sphereVerts.push(sphereY[v2]);          sphereVerts.push(sphereZ[v2++])        }        v1 = v11;        v2 = voff;        sphereVerts.push(sphereX[v1]);        sphereVerts.push(sphereY[v1]);        sphereVerts.push(sphereZ[v1]);        sphereVerts.push(sphereX[v2]);        sphereVerts.push(sphereY[v2]);        sphereVerts.push(sphereZ[v2])      }      for (i = 0; i < sphereDetailU; i++) {        v2 = voff + i;        sphereVerts.push(sphereX[v2]);        sphereVerts.push(sphereY[v2]);        sphereVerts.push(sphereZ[v2]);        sphereVerts.push(0);        sphereVerts.push(1);        sphereVerts.push(0)      }      sphereVerts.push(sphereX[voff]);      sphereVerts.push(sphereY[voff]);      sphereVerts.push(sphereZ[voff]);      sphereVerts.push(0);      sphereVerts.push(1);      sphereVerts.push(0);      curContext.bindBuffer(curContext.ARRAY_BUFFER, sphereBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(sphereVerts), curContext.STATIC_DRAW)    };    p.sphereDetail = function(ures, vres) {      var i;      if (arguments.length === 1) ures = vres = arguments[0];      if (ures < 3) ures = 3;      if (vres < 2) vres = 2;      if (ures === sphereDetailU && vres === sphereDetailV) return;      var delta = 720 / ures;      var cx = new Float32Array(ures);      var cz = new Float32Array(ures);      for (i = 0; i < ures; i++) {        cx[i] = cosLUT[i * delta % 720 | 0];        cz[i] = sinLUT[i * delta % 720 | 0]      }      var vertCount = ures * (vres - 1) + 2;      var currVert = 0;      sphereX = new Float32Array(vertCount);      sphereY = new Float32Array(vertCount);      sphereZ = new Float32Array(vertCount);      var angle_step = 720 * 0.5 / vres;      var angle = angle_step;      for (i = 1; i < vres; i++) {        var curradius = sinLUT[angle % 720 | 0];        var currY = -cosLUT[angle % 720 | 0];        for (var j = 0; j < ures; j++) {          sphereX[currVert] = cx[j] * curradius;          sphereY[currVert] = currY;          sphereZ[currVert++] = cz[j] * curradius        }        angle += angle_step      }      sphereDetailU = ures;      sphereDetailV = vres;      initSphere()    };    Drawing2D.prototype.sphere = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.sphere = function() {      var sRad = arguments[0];      if (sphereDetailU < 3 || sphereDetailV < 2) p.sphereDetail(30);      var model = new PMatrix3D;      model.scale(sRad, sRad, sRad);      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      if (doFill) {        if (lightCount > 0) {          var v = new PMatrix3D;          v.set(view);          var m = new PMatrix3D;          m.set(model);          v.mult(m);          var normalMatrix = new PMatrix3D;          normalMatrix.set(v);          normalMatrix.invert();          normalMatrix.transpose();          uniformMatrix("uNormalTransform3d", programObject3D, "uNormalTransform", false, normalMatrix.array());          vertexAttribPointer("aNormal3d", programObject3D, "aNormal", 3, sphereBuffer)        } else disableVertexAttribPointer("aNormal3d", programObject3D, "aNormal");        curContext.useProgram(programObject3D);        disableVertexAttribPointer("aTexture3d", programObject3D, "aTexture");        uniformMatrix("uModel3d", programObject3D, "uModel", false, model.array());        uniformMatrix("uView3d", programObject3D, "uView", false, view.array());        vertexAttribPointer("aVertex3d", programObject3D, "aVertex", 3, sphereBuffer);        disableVertexAttribPointer("aColor3d", programObject3D, "aColor");        curContext.enable(curContext.POLYGON_OFFSET_FILL);        curContext.polygonOffset(1, 1);        uniformf("uColor3d", programObject3D, "uColor", fillStyle);        curContext.drawArrays(curContext.TRIANGLE_STRIP, 0, sphereVerts.length / 3);        curContext.disable(curContext.POLYGON_OFFSET_FILL)      }      if (lineWidth > 0 && doStroke) {        curContext.useProgram(programObject2D);        uniformMatrix("uModel2d", programObject2D, "uModel", false, model.array());        uniformMatrix("uView2d", programObject2D, "uView", false, view.array());        vertexAttribPointer("aVertex2d", programObject2D, "aVertex", 3, sphereBuffer);        disableVertexAttribPointer("aTextureCoord2d", programObject2D, "aTextureCoord");        uniformf("uColor2d", programObject2D, "uColor", strokeStyle);        uniformi("uIsDrawingText", programObject2D, "uIsDrawingText", false);        curContext.drawArrays(curContext.LINE_STRIP, 0, sphereVerts.length / 3)      }    };    p.modelX = function(x, y, z) {      var mv = modelView.array();      var ci = cameraInv.array();      var ax = mv[0] * x + mv[1] * y + mv[2] * z + mv[3];      var ay = mv[4] * x + mv[5] * y + mv[6] * z + mv[7];      var az = mv[8] * x + mv[9] * y + mv[10] * z + mv[11];      var aw = mv[12] * x + mv[13] * y + mv[14] * z + mv[15];      var ox = ci[0] * ax + ci[1] * ay + ci[2] * az + ci[3] * aw;      var ow = ci[12] * ax + ci[13] * ay + ci[14] * az + ci[15] * aw;      return ow !== 0 ? ox / ow : ox    };    p.modelY = function(x, y, z) {      var mv = modelView.array();      var ci = cameraInv.array();      var ax = mv[0] * x + mv[1] * y + mv[2] * z + mv[3];      var ay = mv[4] * x + mv[5] * y + mv[6] * z + mv[7];      var az = mv[8] * x + mv[9] * y + mv[10] * z + mv[11];      var aw = mv[12] * x + mv[13] * y + mv[14] * z + mv[15];      var oy = ci[4] * ax + ci[5] * ay + ci[6] * az + ci[7] * aw;      var ow = ci[12] * ax + ci[13] * ay + ci[14] * az + ci[15] * aw;      return ow !== 0 ? oy / ow : oy    };    p.modelZ = function(x, y, z) {      var mv = modelView.array();      var ci = cameraInv.array();      var ax = mv[0] * x + mv[1] * y + mv[2] * z + mv[3];      var ay = mv[4] * x + mv[5] * y + mv[6] * z + mv[7];      var az = mv[8] * x + mv[9] * y + mv[10] * z + mv[11];      var aw = mv[12] * x + mv[13] * y + mv[14] * z + mv[15];      var oz = ci[8] * ax + ci[9] * ay + ci[10] * az + ci[11] * aw;      var ow = ci[12] * ax + ci[13] * ay + ci[14] * az + ci[15] * aw;      return ow !== 0 ? oz / ow : oz    };    Drawing2D.prototype.ambient = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.ambient = function(v1, v2, v3) {      curContext.useProgram(programObject3D);      uniformi("uUsingMat3d", programObject3D, "uUsingMat", true);      var col = p.color(v1, v2, v3);      uniformf("uMaterialAmbient3d", programObject3D, "uMaterialAmbient", p.color.toGLArray(col).slice(0, 3))    };    Drawing2D.prototype.emissive = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.emissive = function(v1, v2, v3) {      curContext.useProgram(programObject3D);      uniformi("uUsingMat3d", programObject3D, "uUsingMat", true);      var col = p.color(v1, v2, v3);      uniformf("uMaterialEmissive3d", programObject3D, "uMaterialEmissive", p.color.toGLArray(col).slice(0, 3))    };    Drawing2D.prototype.shininess = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.shininess = function(shine) {      curContext.useProgram(programObject3D);      uniformi("uUsingMat3d", programObject3D, "uUsingMat", true);      uniformf("uShininess3d", programObject3D, "uShininess", shine)    };    Drawing2D.prototype.specular = DrawingShared.prototype.a3DOnlyFunction;    Drawing3D.prototype.specular = function(v1, v2, v3) {      curContext.useProgram(programObject3D);      uniformi("uUsingMat3d", programObject3D, "uUsingMat", true);      var col = p.color(v1, v2, v3);      uniformf("uMaterialSpecular3d", programObject3D, "uMaterialSpecular", p.color.toGLArray(col).slice(0, 3))    };    p.screenX = function(x, y, z) {      var mv = modelView.array();      if (mv.length === 16) {        var ax = mv[0] * x + mv[1] * y + mv[2] * z + mv[3];        var ay = mv[4] * x + mv[5] * y + mv[6] * z + mv[7];        var az = mv[8] * x + mv[9] * y + mv[10] * z + mv[11];        var aw = mv[12] * x + mv[13] * y + mv[14] * z + mv[15];        var pj = projection.array();        var ox = pj[0] * ax + pj[1] * ay + pj[2] * az + pj[3] * aw;        var ow = pj[12] * ax + pj[13] * ay + pj[14] * az + pj[15] * aw;        if (ow !== 0) ox /= ow;        return p.width * (1 + ox) / 2      }      return modelView.multX(x, y)    };    p.screenY = function screenY(x, y, z) {      var mv = modelView.array();      if (mv.length === 16) {        var ax = mv[0] * x + mv[1] * y + mv[2] * z + mv[3];        var ay = mv[4] * x + mv[5] * y + mv[6] * z + mv[7];        var az = mv[8] * x + mv[9] * y + mv[10] * z + mv[11];        var aw = mv[12] * x + mv[13] * y + mv[14] * z + mv[15];        var pj = projection.array();        var oy = pj[4] * ax + pj[5] * ay + pj[6] * az + pj[7] * aw;        var ow = pj[12] * ax + pj[13] * ay + pj[14] * az + pj[15] * aw;        if (ow !== 0) oy /= ow;        return p.height * (1 + oy) / 2      }      return modelView.multY(x, y)    };    p.screenZ = function screenZ(x, y, z) {      var mv = modelView.array();      if (mv.length !== 16) return 0;      var pj = projection.array();      var ax = mv[0] * x + mv[1] * y + mv[2] * z + mv[3];      var ay = mv[4] * x + mv[5] * y + mv[6] * z + mv[7];      var az = mv[8] * x + mv[9] * y + mv[10] * z + mv[11];      var aw = mv[12] * x + mv[13] * y + mv[14] * z + mv[15];      var oz = pj[8] * ax + pj[9] * ay + pj[10] * az + pj[11] * aw;      var ow = pj[12] * ax + pj[13] * ay + pj[14] * az + pj[15] * aw;      if (ow !== 0) oz /= ow;      return (oz + 1) / 2    };    DrawingShared.prototype.fill = function() {      var color = p.color(arguments[0], arguments[1], arguments[2], arguments[3]);      if (color === currentFillColor && doFill) return;      doFill = true;      currentFillColor = color    };    Drawing2D.prototype.fill = function() {      DrawingShared.prototype.fill.apply(this, arguments);      isFillDirty = true    };    Drawing3D.prototype.fill = function() {      DrawingShared.prototype.fill.apply(this, arguments);      fillStyle = p.color.toGLArray(currentFillColor)    };    function executeContextFill() {      if (doFill) {        if (isFillDirty) {          curContext.fillStyle = p.color.toString(currentFillColor);          isFillDirty = false        }        curContext.fill()      }    }    p.noFill = function() {      doFill = false    };    DrawingShared.prototype.stroke = function() {      var color = p.color(arguments[0], arguments[1], arguments[2], arguments[3]);      if (color === currentStrokeColor && doStroke) return;      doStroke = true;      currentStrokeColor = color    };    Drawing2D.prototype.stroke = function() {      DrawingShared.prototype.stroke.apply(this, arguments);      isStrokeDirty = true    };    Drawing3D.prototype.stroke = function() {      DrawingShared.prototype.stroke.apply(this, arguments);      strokeStyle = p.color.toGLArray(currentStrokeColor)    };    function executeContextStroke() {      if (doStroke) {        if (isStrokeDirty) {          curContext.strokeStyle = p.color.toString(currentStrokeColor);          isStrokeDirty = false        }        curContext.stroke()      }    }    p.noStroke = function() {      doStroke = false    };    DrawingShared.prototype.strokeWeight = function(w) {      lineWidth = w    };    Drawing2D.prototype.strokeWeight = function(w) {      DrawingShared.prototype.strokeWeight.apply(this, arguments);      curContext.lineWidth = w    };    Drawing3D.prototype.strokeWeight = function(w) {      DrawingShared.prototype.strokeWeight.apply(this, arguments);      curContext.useProgram(programObject2D);      uniformf("pointSize2d", programObject2D, "uPointSize", w);      curContext.useProgram(programObjectUnlitShape);      uniformf("pointSizeUnlitShape", programObjectUnlitShape, "uPointSize", w);      curContext.lineWidth(w)    };    p.strokeCap = function(value) {      drawing.$ensureContext().lineCap = value    };    p.strokeJoin = function(value) {      drawing.$ensureContext().lineJoin = value    };    Drawing2D.prototype.smooth = function() {      renderSmooth = true;      var style = curElement.style;      style.setProperty("image-rendering", "optimizeQuality", "important");      style.setProperty("-ms-interpolation-mode", "bicubic", "important");      if (curContext.hasOwnProperty("mozImageSmoothingEnabled")) curContext.mozImageSmoothingEnabled = true    };    Drawing3D.prototype.smooth = function() {      renderSmooth = true    };    Drawing2D.prototype.noSmooth = function() {      renderSmooth = false;      var style = curElement.style;      style.setProperty("image-rendering", "optimizeSpeed", "important");      style.setProperty("image-rendering", "-moz-crisp-edges", "important");      style.setProperty("image-rendering", "-webkit-optimize-contrast", "important");      style.setProperty("image-rendering", "optimize-contrast", "important");      style.setProperty("-ms-interpolation-mode", "nearest-neighbor", "important");      if (curContext.hasOwnProperty("mozImageSmoothingEnabled")) curContext.mozImageSmoothingEnabled = false    };    Drawing3D.prototype.noSmooth = function() {      renderSmooth = false    };    Drawing2D.prototype.point = function(x, y) {      if (!doStroke) return;      x = Math.round(x);      y = Math.round(y);      curContext.fillStyle = p.color.toString(currentStrokeColor);      isFillDirty = true;      if (lineWidth > 1) {        curContext.beginPath();        curContext.arc(x, y, lineWidth / 2, 0, 6.283185307179586, false);        curContext.fill()      } else curContext.fillRect(x, y, 1, 1)    };    Drawing3D.prototype.point = function(x, y, z) {      var model = new PMatrix3D;      model.translate(x, y, z || 0);      model.transpose();      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      curContext.useProgram(programObject2D);      uniformMatrix("uModel2d", programObject2D, "uModel", false, model.array());      uniformMatrix("uView2d", programObject2D, "uView", false, view.array());      if (lineWidth > 0 && doStroke) {        uniformf("uColor2d", programObject2D, "uColor", strokeStyle);        uniformi("uIsDrawingText2d", programObject2D, "uIsDrawingText", false);        uniformi("uSmooth2d", programObject2D, "uSmooth", renderSmooth);        vertexAttribPointer("aVertex2d", programObject2D, "aVertex", 3, pointBuffer);        disableVertexAttribPointer("aTextureCoord2d", programObject2D, "aTextureCoord");        curContext.drawArrays(curContext.POINTS, 0, 1)      }    };    p.beginShape = function(type) {      curShape = type;      curvePoints = [];      vertArray = []    };    Drawing2D.prototype.vertex = function(x, y, moveTo) {      var vert = [];      if (firstVert) firstVert = false;      vert["isVert"] = true;      vert[0] = x;      vert[1] = y;      vert[2] = 0;      vert[3] = 0;      vert[4] = 0;      vert[5] = currentFillColor;      vert[6] = currentStrokeColor;      vertArray.push(vert);      if (moveTo) vertArray[vertArray.length - 1]["moveTo"] = moveTo    };    Drawing3D.prototype.vertex = function(x, y, z, u, v) {      var vert = [];      if (firstVert) firstVert = false;      vert["isVert"] = true;      if (v === undef && usingTexture) {        v = u;        u = z;        z = 0      }      if (u !== undef && v !== undef) {        if (curTextureMode === 2) {          u /= curTexture.width;          v /= curTexture.height        }        u = u > 1 ? 1 : u;        u = u < 0 ? 0 : u;        v = v > 1 ? 1 : v;        v = v < 0 ? 0 : v      }      vert[0] = x;      vert[1] = y;      vert[2] = z || 0;      vert[3] = u || 0;      vert[4] = v || 0;      vert[5] = fillStyle[0];      vert[6] = fillStyle[1];      vert[7] = fillStyle[2];      vert[8] = fillStyle[3];      vert[9] = strokeStyle[0];      vert[10] = strokeStyle[1];      vert[11] = strokeStyle[2];      vert[12] = strokeStyle[3];      vert[13] = normalX;      vert[14] = normalY;      vert[15] = normalZ;      vertArray.push(vert)    };    var point3D = function(vArray, cArray) {      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      curContext.useProgram(programObjectUnlitShape);      uniformMatrix("uViewUS", programObjectUnlitShape, "uView", false, view.array());      uniformi("uSmoothUS", programObjectUnlitShape, "uSmooth", renderSmooth);      vertexAttribPointer("aVertexUS", programObjectUnlitShape, "aVertex", 3, pointBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(vArray), curContext.STREAM_DRAW);      vertexAttribPointer("aColorUS", programObjectUnlitShape, "aColor", 4, fillColorBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(cArray), curContext.STREAM_DRAW);      curContext.drawArrays(curContext.POINTS, 0, vArray.length / 3)    };    var line3D = function(vArray, mode, cArray) {      var ctxMode;      if (mode === "LINES") ctxMode = curContext.LINES;      else if (mode === "LINE_LOOP") ctxMode = curContext.LINE_LOOP;      else ctxMode = curContext.LINE_STRIP;      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      curContext.useProgram(programObjectUnlitShape);      uniformMatrix("uViewUS", programObjectUnlitShape, "uView", false, view.array());      vertexAttribPointer("aVertexUS", programObjectUnlitShape, "aVertex", 3, lineBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(vArray), curContext.STREAM_DRAW);      vertexAttribPointer("aColorUS", programObjectUnlitShape, "aColor", 4, strokeColorBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(cArray), curContext.STREAM_DRAW);      curContext.drawArrays(ctxMode, 0, vArray.length / 3)    };    var fill3D = function(vArray, mode, cArray, tArray) {      var ctxMode;      if (mode === "TRIANGLES") ctxMode = curContext.TRIANGLES;      else if (mode === "TRIANGLE_FAN") ctxMode = curContext.TRIANGLE_FAN;      else ctxMode = curContext.TRIANGLE_STRIP;      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      curContext.useProgram(programObject3D);      uniformMatrix("model3d", programObject3D, "uModel", false, [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]);      uniformMatrix("view3d", programObject3D, "uView", false, view.array());      curContext.enable(curContext.POLYGON_OFFSET_FILL);      curContext.polygonOffset(1, 1);      uniformf("color3d", programObject3D, "uColor", [-1, 0, 0, 0]);      vertexAttribPointer("vertex3d", programObject3D, "aVertex", 3, fillBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(vArray), curContext.STREAM_DRAW);      if (usingTexture && curTint !== null) curTint3d(cArray);      vertexAttribPointer("aColor3d", programObject3D, "aColor", 4, fillColorBuffer);      curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(cArray), curContext.STREAM_DRAW);      disableVertexAttribPointer("aNormal3d", programObject3D, "aNormal");      if (usingTexture) {        uniformi("uUsingTexture3d", programObject3D, "uUsingTexture", usingTexture);        vertexAttribPointer("aTexture3d", programObject3D, "aTexture", 2, shapeTexVBO);        curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(tArray), curContext.STREAM_DRAW)      }      curContext.drawArrays(ctxMode, 0, vArray.length / 3);      curContext.disable(curContext.POLYGON_OFFSET_FILL)    };    function fillStrokeClose() {      executeContextFill();      executeContextStroke();      curContext.closePath()    }    Drawing2D.prototype.endShape = function(mode) {      if (vertArray.length === 0) return;      var closeShape = mode === 2;      if (closeShape) vertArray.push(vertArray[0]);      var lineVertArray = [];      var fillVertArray = [];      var colorVertArray = [];      var strokeVertArray = [];      var texVertArray = [];      var cachedVertArray;      firstVert = true;      var i, j, k;      var vertArrayLength = vertArray.length;      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        for (j = 0; j < 3; j++) fillVertArray.push(cachedVertArray[j])      }      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        for (j = 5; j < 9; j++) colorVertArray.push(cachedVertArray[j])      }      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        for (j = 9; j < 13; j++) strokeVertArray.push(cachedVertArray[j])      }      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        texVertArray.push(cachedVertArray[3]);        texVertArray.push(cachedVertArray[4])      }      if (isCurve && (curShape === 20 || curShape === undef)) {        if (vertArrayLength > 3) {          var b = [],            s = 1 - curTightness;          curContext.beginPath();          curContext.moveTo(vertArray[1][0], vertArray[1][1]);          for (i = 1; i + 2 < vertArrayLength; i++) {            cachedVertArray = vertArray[i];            b[0] = [cachedVertArray[0], cachedVertArray[1]];            b[1] = [cachedVertArray[0] + (s * vertArray[i + 1][0] - s * vertArray[i - 1][0]) / 6, cachedVertArray[1] + (s * vertArray[i + 1][1] - s * vertArray[i - 1][1]) / 6];            b[2] = [vertArray[i + 1][0] + (s * vertArray[i][0] - s * vertArray[i + 2][0]) / 6, vertArray[i + 1][1] + (s * vertArray[i][1] - s * vertArray[i + 2][1]) / 6];            b[3] = [vertArray[i + 1][0], vertArray[i + 1][1]];            curContext.bezierCurveTo(b[1][0], b[1][1], b[2][0], b[2][1], b[3][0], b[3][1])          }          fillStrokeClose()        }      } else if (isBezier && (curShape === 20 || curShape === undef)) {        curContext.beginPath();        for (i = 0; i < vertArrayLength; i++) {          cachedVertArray = vertArray[i];          if (vertArray[i]["isVert"]) if (vertArray[i]["moveTo"]) curContext.moveTo(cachedVertArray[0], cachedVertArray[1]);          else curContext.lineTo(cachedVertArray[0], cachedVertArray[1]);          else curContext.bezierCurveTo(vertArray[i][0], vertArray[i][1], vertArray[i][2], vertArray[i][3], vertArray[i][4], vertArray[i][5])        }        fillStrokeClose()      } else if (curShape === 2) for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        if (doStroke) p.stroke(cachedVertArray[6]);        p.point(cachedVertArray[0], cachedVertArray[1])      } else if (curShape === 4) for (i = 0; i + 1 < vertArrayLength; i += 2) {        cachedVertArray = vertArray[i];        if (doStroke) p.stroke(vertArray[i + 1][6]);        p.line(cachedVertArray[0], cachedVertArray[1], vertArray[i + 1][0], vertArray[i + 1][1])      } else if (curShape === 9) for (i = 0; i + 2 < vertArrayLength; i += 3) {        cachedVertArray = vertArray[i];        curContext.beginPath();        curContext.moveTo(cachedVertArray[0], cachedVertArray[1]);        curContext.lineTo(vertArray[i + 1][0], vertArray[i + 1][1]);        curContext.lineTo(vertArray[i + 2][0], vertArray[i + 2][1]);        curContext.lineTo(cachedVertArray[0], cachedVertArray[1]);        if (doFill) {          p.fill(vertArray[i + 2][5]);          executeContextFill()        }        if (doStroke) {          p.stroke(vertArray[i + 2][6]);          executeContextStroke()        }        curContext.closePath()      } else if (curShape === 10) for (i = 0; i + 1 < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        curContext.beginPath();        curContext.moveTo(vertArray[i + 1][0], vertArray[i + 1][1]);        curContext.lineTo(cachedVertArray[0], cachedVertArray[1]);        if (doStroke) p.stroke(vertArray[i + 1][6]);        if (doFill) p.fill(vertArray[i + 1][5]);        if (i + 2 < vertArrayLength) {          curContext.lineTo(vertArray[i + 2][0], vertArray[i + 2][1]);          if (doStroke) p.stroke(vertArray[i + 2][6]);          if (doFill) p.fill(vertArray[i + 2][5])        }        fillStrokeClose()      } else if (curShape === 11) {        if (vertArrayLength > 2) {          curContext.beginPath();          curContext.moveTo(vertArray[0][0], vertArray[0][1]);          curContext.lineTo(vertArray[1][0], vertArray[1][1]);          curContext.lineTo(vertArray[2][0], vertArray[2][1]);          if (doFill) {            p.fill(vertArray[2][5]);            executeContextFill()          }          if (doStroke) {            p.stroke(vertArray[2][6]);            executeContextStroke()          }          curContext.closePath();          for (i = 3; i < vertArrayLength; i++) {            cachedVertArray = vertArray[i];            curContext.beginPath();            curContext.moveTo(vertArray[0][0], vertArray[0][1]);            curContext.lineTo(vertArray[i - 1][0], vertArray[i - 1][1]);            curContext.lineTo(cachedVertArray[0], cachedVertArray[1]);            if (doFill) {              p.fill(cachedVertArray[5]);              executeContextFill()            }            if (doStroke) {              p.stroke(cachedVertArray[6]);              executeContextStroke()            }            curContext.closePath()          }        }      } else if (curShape === 16) for (i = 0; i + 3 < vertArrayLength; i += 4) {        cachedVertArray = vertArray[i];        curContext.beginPath();        curContext.moveTo(cachedVertArray[0], cachedVertArray[1]);        for (j = 1; j < 4; j++) curContext.lineTo(vertArray[i + j][0], vertArray[i + j][1]);        curContext.lineTo(cachedVertArray[0], cachedVertArray[1]);        if (doFill) {          p.fill(vertArray[i + 3][5]);          executeContextFill()        }        if (doStroke) {          p.stroke(vertArray[i + 3][6]);          executeContextStroke()        }        curContext.closePath()      } else if (curShape === 17) {        if (vertArrayLength > 3) for (i = 0; i + 1 < vertArrayLength; i += 2) {          cachedVertArray = vertArray[i];          curContext.beginPath();          if (i + 3 < vertArrayLength) {            curContext.moveTo(vertArray[i + 2][0], vertArray[i + 2][1]);            curContext.lineTo(cachedVertArray[0], cachedVertArray[1]);            curContext.lineTo(vertArray[i + 1][0], vertArray[i + 1][1]);            curContext.lineTo(vertArray[i + 3][0], vertArray[i + 3][1]);            if (doFill) p.fill(vertArray[i + 3][5]);            if (doStroke) p.stroke(vertArray[i + 3][6])          } else {            curContext.moveTo(cachedVertArray[0], cachedVertArray[1]);            curContext.lineTo(vertArray[i + 1][0], vertArray[i + 1][1])          }          fillStrokeClose()        }      } else {        curContext.beginPath();        curContext.moveTo(vertArray[0][0], vertArray[0][1]);        for (i = 1; i < vertArrayLength; i++) {          cachedVertArray = vertArray[i];          if (cachedVertArray["isVert"]) if (cachedVertArray["moveTo"]) curContext.moveTo(cachedVertArray[0], cachedVertArray[1]);          else curContext.lineTo(cachedVertArray[0], cachedVertArray[1])        }        fillStrokeClose()      }      isCurve = false;      isBezier = false;      curveVertArray = [];      curveVertCount = 0;      if (closeShape) vertArray.pop()    };    Drawing3D.prototype.endShape = function(mode) {      if (vertArray.length === 0) return;      var closeShape = mode === 2;      var lineVertArray = [];      var fillVertArray = [];      var colorVertArray = [];      var strokeVertArray = [];      var texVertArray = [];      var cachedVertArray;      firstVert = true;      var i, j, k;      var vertArrayLength = vertArray.length;      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        for (j = 0; j < 3; j++) fillVertArray.push(cachedVertArray[j])      }      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        for (j = 5; j < 9; j++) colorVertArray.push(cachedVertArray[j])      }      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        for (j = 9; j < 13; j++) strokeVertArray.push(cachedVertArray[j])      }      for (i = 0; i < vertArrayLength; i++) {        cachedVertArray = vertArray[i];        texVertArray.push(cachedVertArray[3]);        texVertArray.push(cachedVertArray[4])      }      if (closeShape) {        fillVertArray.push(vertArray[0][0]);        fillVertArray.push(vertArray[0][1]);        fillVertArray.push(vertArray[0][2]);        for (i = 5; i < 9; i++) colorVertArray.push(vertArray[0][i]);        for (i = 9; i < 13; i++) strokeVertArray.push(vertArray[0][i]);        texVertArray.push(vertArray[0][3]);        texVertArray.push(vertArray[0][4])      }      if (isCurve && (curShape === 20 || curShape === undef)) {        lineVertArray = fillVertArray;        if (doStroke) line3D(lineVertArray, null, strokeVertArray);        if (doFill) fill3D(fillVertArray, null, colorVertArray)      } else if (isBezier && (curShape === 20 || curShape === undef)) {        lineVertArray = fillVertArray;        lineVertArray.splice(lineVertArray.length - 3);        strokeVertArray.splice(strokeVertArray.length - 4);        if (doStroke) line3D(lineVertArray, null, strokeVertArray);        if (doFill) fill3D(fillVertArray, "TRIANGLES", colorVertArray)      } else {        if (curShape === 2) {          for (i = 0; i < vertArrayLength; i++) {            cachedVertArray = vertArray[i];            for (j = 0; j < 3; j++) lineVertArray.push(cachedVertArray[j])          }          point3D(lineVertArray, strokeVertArray)        } else if (curShape === 4) {          for (i = 0; i < vertArrayLength; i++) {            cachedVertArray = vertArray[i];            for (j = 0; j < 3; j++) lineVertArray.push(cachedVertArray[j])          }          for (i = 0; i < vertArrayLength; i++) {            cachedVertArray = vertArray[i];            for (j = 5; j < 9; j++) colorVertArray.push(cachedVertArray[j])          }          line3D(lineVertArray, "LINES", strokeVertArray)        } else if (curShape === 9) {          if (vertArrayLength > 2) for (i = 0; i + 2 < vertArrayLength; i += 3) {            fillVertArray = [];            texVertArray = [];            lineVertArray = [];            colorVertArray = [];            strokeVertArray = [];            for (j = 0; j < 3; j++) for (k = 0; k < 3; k++) {              lineVertArray.push(vertArray[i + j][k]);              fillVertArray.push(vertArray[i + j][k])            }            for (j = 0; j < 3; j++) for (k = 3; k < 5; k++) texVertArray.push(vertArray[i + j][k]);            for (j = 0; j < 3; j++) for (k = 5; k < 9; k++) {              colorVertArray.push(vertArray[i + j][k]);              strokeVertArray.push(vertArray[i + j][k + 4])            }            if (doStroke) line3D(lineVertArray, "LINE_LOOP", strokeVertArray);            if (doFill || usingTexture) fill3D(fillVertArray, "TRIANGLES", colorVertArray, texVertArray)          }        } else if (curShape === 10) {          if (vertArrayLength > 2) for (i = 0; i + 2 < vertArrayLength; i++) {            lineVertArray = [];            fillVertArray = [];            strokeVertArray = [];            colorVertArray = [];            texVertArray = [];            for (j = 0; j < 3; j++) for (k = 0; k < 3; k++) {              lineVertArray.push(vertArray[i + j][k]);              fillVertArray.push(vertArray[i + j][k])            }            for (j = 0; j < 3; j++) for (k = 3; k < 5; k++) texVertArray.push(vertArray[i + j][k]);            for (j = 0; j < 3; j++) for (k = 5; k < 9; k++) {              strokeVertArray.push(vertArray[i + j][k + 4]);              colorVertArray.push(vertArray[i + j][k])            }            if (doFill || usingTexture) fill3D(fillVertArray, "TRIANGLE_STRIP", colorVertArray, texVertArray);            if (doStroke) line3D(lineVertArray, "LINE_LOOP", strokeVertArray)          }        } else if (curShape === 11) {          if (vertArrayLength > 2) {            for (i = 0; i < 3; i++) {              cachedVertArray = vertArray[i];              for (j = 0; j < 3; j++) lineVertArray.push(cachedVertArray[j])            }            for (i = 0; i < 3; i++) {              cachedVertArray = vertArray[i];              for (j = 9; j < 13; j++) strokeVertArray.push(cachedVertArray[j])            }            if (doStroke) line3D(lineVertArray, "LINE_LOOP", strokeVertArray);            for (i = 2; i + 1 < vertArrayLength; i++) {              lineVertArray = [];              strokeVertArray = [];              lineVertArray.push(vertArray[0][0]);              lineVertArray.push(vertArray[0][1]);              lineVertArray.push(vertArray[0][2]);              strokeVertArray.push(vertArray[0][9]);              strokeVertArray.push(vertArray[0][10]);              strokeVertArray.push(vertArray[0][11]);              strokeVertArray.push(vertArray[0][12]);              for (j = 0; j < 2; j++) for (k = 0; k < 3; k++) lineVertArray.push(vertArray[i + j][k]);              for (j = 0; j < 2; j++) for (k = 9; k < 13; k++) strokeVertArray.push(vertArray[i + j][k]);              if (doStroke) line3D(lineVertArray, "LINE_STRIP", strokeVertArray)            }            if (doFill || usingTexture) fill3D(fillVertArray, "TRIANGLE_FAN", colorVertArray, texVertArray)          }        } else if (curShape === 16) for (i = 0; i + 3 < vertArrayLength; i += 4) {          lineVertArray = [];          for (j = 0; j < 4; j++) {            cachedVertArray = vertArray[i + j];            for (k = 0; k < 3; k++) lineVertArray.push(cachedVertArray[k])          }          if (doStroke) line3D(lineVertArray, "LINE_LOOP", strokeVertArray);          if (doFill) {            fillVertArray = [];            colorVertArray = [];            texVertArray = [];            for (j = 0; j < 3; j++) fillVertArray.push(vertArray[i][j]);            for (j = 5; j < 9; j++) colorVertArray.push(vertArray[i][j]);            for (j = 0; j < 3; j++) fillVertArray.push(vertArray[i + 1][j]);            for (j = 5; j < 9; j++) colorVertArray.push(vertArray[i + 1][j]);            for (j = 0; j < 3; j++) fillVertArray.push(vertArray[i + 3][j]);            for (j = 5; j < 9; j++) colorVertArray.push(vertArray[i + 3][j]);            for (j = 0; j < 3; j++) fillVertArray.push(vertArray[i + 2][j]);            for (j = 5; j < 9; j++) colorVertArray.push(vertArray[i + 2][j]);            if (usingTexture) {              texVertArray.push(vertArray[i + 0][3]);              texVertArray.push(vertArray[i + 0][4]);              texVertArray.push(vertArray[i + 1][3]);              texVertArray.push(vertArray[i + 1][4]);              texVertArray.push(vertArray[i + 3][3]);              texVertArray.push(vertArray[i + 3][4]);              texVertArray.push(vertArray[i + 2][3]);              texVertArray.push(vertArray[i + 2][4])            }            fill3D(fillVertArray, "TRIANGLE_STRIP", colorVertArray, texVertArray)          }        } else if (curShape === 17) {          var tempArray = [];          if (vertArrayLength > 3) {            for (i = 0; i < 2; i++) {              cachedVertArray = vertArray[i];              for (j = 0; j < 3; j++) lineVertArray.push(cachedVertArray[j])            }            for (i = 0; i < 2; i++) {              cachedVertArray = vertArray[i];              for (j = 9; j < 13; j++) strokeVertArray.push(cachedVertArray[j])            }            line3D(lineVertArray, "LINE_STRIP", strokeVertArray);            if (vertArrayLength > 4 && vertArrayLength % 2 > 0) {              tempArray = fillVertArray.splice(fillVertArray.length - 3);              vertArray.pop()            }            for (i = 0; i + 3 < vertArrayLength; i += 2) {              lineVertArray = [];              strokeVertArray = [];              for (j = 0; j < 3; j++) lineVertArray.push(vertArray[i + 1][j]);              for (j = 0; j < 3; j++) lineVertArray.push(vertArray[i + 3][j]);              for (j = 0; j < 3; j++) lineVertArray.push(vertArray[i + 2][j]);              for (j = 0; j < 3; j++) lineVertArray.push(vertArray[i + 0][j]);              for (j = 9; j < 13; j++) strokeVertArray.push(vertArray[i + 1][j]);              for (j = 9; j < 13; j++) strokeVertArray.push(vertArray[i + 3][j]);              for (j = 9; j < 13; j++) strokeVertArray.push(vertArray[i + 2][j]);              for (j = 9; j < 13; j++) strokeVertArray.push(vertArray[i + 0][j]);              if (doStroke) line3D(lineVertArray, "LINE_STRIP", strokeVertArray)            }            if (doFill || usingTexture) fill3D(fillVertArray, "TRIANGLE_LIST", colorVertArray, texVertArray)          }        } else if (vertArrayLength === 1) {          for (j = 0; j < 3; j++) lineVertArray.push(vertArray[0][j]);          for (j = 9; j < 13; j++) strokeVertArray.push(vertArray[0][j]);          point3D(lineVertArray, strokeVertArray)        } else {          for (i = 0; i < vertArrayLength; i++) {            cachedVertArray = vertArray[i];            for (j = 0; j < 3; j++) lineVertArray.push(cachedVertArray[j]);            for (j = 5; j < 9; j++) strokeVertArray.push(cachedVertArray[j])          }          if (doStroke && closeShape) line3D(lineVertArray, "LINE_LOOP", strokeVertArray);          else if (doStroke && !closeShape) line3D(lineVertArray, "LINE_STRIP", strokeVertArray);          if (doFill || usingTexture) fill3D(fillVertArray, "TRIANGLE_FAN", colorVertArray, texVertArray)        }        usingTexture = false;        curContext.useProgram(programObject3D);        uniformi("usingTexture3d", programObject3D, "uUsingTexture", usingTexture)      }      isCurve = false;      isBezier = false;      curveVertArray = [];      curveVertCount = 0    };    var splineForward = function(segments, matrix) {      var f = 1 / segments;      var ff = f * f;      var fff = ff * f;      matrix.set(0, 0, 0, 1, fff, ff, f, 0, 6 * fff, 2 * ff, 0, 0, 6 * fff, 0, 0, 0)    };    var curveInit = function() {      if (!curveDrawMatrix) {        curveBasisMatrix = new PMatrix3D;        curveDrawMatrix = new PMatrix3D;        curveInited = true      }      var s = curTightness;      curveBasisMatrix.set((s - 1) / 2, (s + 3) / 2, (-3 - s) / 2, (1 - s) / 2, 1 - s, (-5 - s) / 2, s + 2, (s - 1) / 2, (s - 1) / 2, 0, (1 - s) / 2, 0, 0, 1, 0, 0);      splineForward(curveDet, curveDrawMatrix);      if (!bezierBasisInverse) curveToBezierMatrix = new PMatrix3D;      curveToBezierMatrix.set(curveBasisMatrix);      curveToBezierMatrix.preApply(bezierBasisInverse);      curveDrawMatrix.apply(curveBasisMatrix)    };    Drawing2D.prototype.bezierVertex = function() {      isBezier = true;      var vert = [];      if (firstVert) throw "vertex() must be used at least once before calling bezierVertex()";      for (var i = 0; i < arguments.length; i++) vert[i] = arguments[i];      vertArray.push(vert);      vertArray[vertArray.length - 1]["isVert"] = false    };    Drawing3D.prototype.bezierVertex = function() {      isBezier = true;      var vert = [];      if (firstVert) throw "vertex() must be used at least once before calling bezierVertex()";      if (arguments.length === 9) {        if (bezierDrawMatrix === undef) bezierDrawMatrix = new PMatrix3D;        var lastPoint = vertArray.length - 1;        splineForward(bezDetail, bezierDrawMatrix);        bezierDrawMatrix.apply(bezierBasisMatrix);        var draw = bezierDrawMatrix.array();        var x1 = vertArray[lastPoint][0],          y1 = vertArray[lastPoint][1],          z1 = vertArray[lastPoint][2];        var xplot1 = draw[4] * x1 + draw[5] * arguments[0] + draw[6] * arguments[3] + draw[7] * arguments[6];        var xplot2 = draw[8] * x1 + draw[9] * arguments[0] + draw[10] * arguments[3] + draw[11] * arguments[6];        var xplot3 = draw[12] * x1 + draw[13] * arguments[0] + draw[14] * arguments[3] + draw[15] * arguments[6];        var yplot1 = draw[4] * y1 + draw[5] * arguments[1] + draw[6] * arguments[4] + draw[7] * arguments[7];        var yplot2 = draw[8] * y1 + draw[9] * arguments[1] + draw[10] * arguments[4] + draw[11] * arguments[7];        var yplot3 = draw[12] * y1 + draw[13] * arguments[1] + draw[14] * arguments[4] + draw[15] * arguments[7];        var zplot1 = draw[4] * z1 + draw[5] * arguments[2] + draw[6] * arguments[5] + draw[7] * arguments[8];        var zplot2 = draw[8] * z1 + draw[9] * arguments[2] + draw[10] * arguments[5] + draw[11] * arguments[8];        var zplot3 = draw[12] * z1 + draw[13] * arguments[2] + draw[14] * arguments[5] + draw[15] * arguments[8];        for (var j = 0; j < bezDetail; j++) {          x1 += xplot1;          xplot1 += xplot2;          xplot2 += xplot3;          y1 += yplot1;          yplot1 += yplot2;          yplot2 += yplot3;          z1 += zplot1;          zplot1 += zplot2;          zplot2 += zplot3;          p.vertex(x1, y1, z1)        }        p.vertex(arguments[6], arguments[7], arguments[8])      }    };    p.texture = function(pimage) {      var curContext = drawing.$ensureContext();      if (pimage.__texture) curContext.bindTexture(curContext.TEXTURE_2D, pimage.__texture);      else if (pimage.localName === "canvas") {        curContext.bindTexture(curContext.TEXTURE_2D, canTex);        curContext.texImage2D(curContext.TEXTURE_2D, 0, curContext.RGBA, curContext.RGBA, curContext.UNSIGNED_BYTE, pimage);        curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_MAG_FILTER, curContext.LINEAR);        curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_MIN_FILTER, curContext.LINEAR);        curContext.generateMipmap(curContext.TEXTURE_2D);        curTexture.width = pimage.width;        curTexture.height = pimage.height      } else {        var texture = curContext.createTexture(),          cvs = document.createElement("canvas"),          cvsTextureCtx = cvs.getContext("2d"),          pot;        if (pimage.width & pimage.width - 1 === 0) cvs.width = pimage.width;        else {          pot = 1;          while (pot < pimage.width) pot *= 2;          cvs.width = pot        }        if (pimage.height & pimage.height - 1 === 0) cvs.height = pimage.height;        else {          pot = 1;          while (pot < pimage.height) pot *= 2;          cvs.height = pot        }        cvsTextureCtx.drawImage(pimage.sourceImg, 0, 0, pimage.width, pimage.height, 0, 0, cvs.width, cvs.height);        curContext.bindTexture(curContext.TEXTURE_2D, texture);        curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_MIN_FILTER, curContext.LINEAR_MIPMAP_LINEAR);        curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_MAG_FILTER, curContext.LINEAR);        curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_WRAP_T, curContext.CLAMP_TO_EDGE);        curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_WRAP_S, curContext.CLAMP_TO_EDGE);        curContext.texImage2D(curContext.TEXTURE_2D, 0, curContext.RGBA, curContext.RGBA, curContext.UNSIGNED_BYTE, cvs);        curContext.generateMipmap(curContext.TEXTURE_2D);        pimage.__texture = texture;        curTexture.width = pimage.width;        curTexture.height = pimage.height      }      usingTexture = true;      curContext.useProgram(programObject3D);      uniformi("usingTexture3d", programObject3D, "uUsingTexture", usingTexture)    };    p.textureMode = function(mode) {      curTextureMode = mode    };    var curveVertexSegment = function(x1, y1, z1, x2, y2, z2, x3, y3, z3, x4, y4, z4) {      var x0 = x2;      var y0 = y2;      var z0 = z2;      var draw = curveDrawMatrix.array();      var xplot1 = draw[4] * x1 + draw[5] * x2 + draw[6] * x3 + draw[7] * x4;      var xplot2 = draw[8] * x1 + draw[9] * x2 + draw[10] * x3 + draw[11] * x4;      var xplot3 = draw[12] * x1 + draw[13] * x2 + draw[14] * x3 + draw[15] * x4;      var yplot1 = draw[4] * y1 + draw[5] * y2 + draw[6] * y3 + draw[7] * y4;      var yplot2 = draw[8] * y1 + draw[9] * y2 + draw[10] * y3 + draw[11] * y4;      var yplot3 = draw[12] * y1 + draw[13] * y2 + draw[14] * y3 + draw[15] * y4;      var zplot1 = draw[4] * z1 + draw[5] * z2 + draw[6] * z3 + draw[7] * z4;      var zplot2 = draw[8] * z1 + draw[9] * z2 + draw[10] * z3 + draw[11] * z4;      var zplot3 = draw[12] * z1 + draw[13] * z2 + draw[14] * z3 + draw[15] * z4;      p.vertex(x0, y0, z0);      for (var j = 0; j < curveDet; j++) {        x0 += xplot1;        xplot1 += xplot2;        xplot2 += xplot3;        y0 += yplot1;        yplot1 += yplot2;        yplot2 += yplot3;        z0 += zplot1;        zplot1 += zplot2;        zplot2 += zplot3;        p.vertex(x0, y0, z0)      }    };    Drawing2D.prototype.curveVertex = function(x, y) {      isCurve = true;      p.vertex(x, y)    };    Drawing3D.prototype.curveVertex = function(x, y, z) {      isCurve = true;      if (!curveInited) curveInit();      var vert = [];      vert[0] = x;      vert[1] = y;      vert[2] = z;      curveVertArray.push(vert);      curveVertCount++;      if (curveVertCount > 3) curveVertexSegment(curveVertArray[curveVertCount - 4][0], curveVertArray[curveVertCount - 4][1], curveVertArray[curveVertCount - 4][2], curveVertArray[curveVertCount - 3][0], curveVertArray[curveVertCount - 3][1], curveVertArray[curveVertCount - 3][2], curveVertArray[curveVertCount - 2][0], curveVertArray[curveVertCount - 2][1], curveVertArray[curveVertCount - 2][2], curveVertArray[curveVertCount - 1][0], curveVertArray[curveVertCount - 1][1], curveVertArray[curveVertCount - 1][2])    };    Drawing2D.prototype.curve = function(x1, y1, x2, y2, x3, y3, x4, y4) {      p.beginShape();      p.curveVertex(x1, y1);      p.curveVertex(x2, y2);      p.curveVertex(x3, y3);      p.curveVertex(x4, y4);      p.endShape()    };    Drawing3D.prototype.curve = function(x1, y1, z1, x2, y2, z2, x3, y3, z3, x4, y4, z4) {      if (z4 !== undef) {        p.beginShape();        p.curveVertex(x1, y1, z1);        p.curveVertex(x2, y2, z2);        p.curveVertex(x3, y3, z3);        p.curveVertex(x4, y4, z4);        p.endShape();        return      }      p.beginShape();      p.curveVertex(x1, y1);      p.curveVertex(z1, x2);      p.curveVertex(y2, z2);      p.curveVertex(x3, y3);      p.endShape()    };    p.curveTightness = function(tightness) {      curTightness = tightness    };    p.curveDetail = function(detail) {      curveDet = detail;      curveInit()    };    p.rectMode = function(aRectMode) {      curRectMode = aRectMode    };    p.imageMode = function(mode) {      switch (mode) {      case 0:        imageModeConvert = imageModeCorner;        break;      case 1:        imageModeConvert = imageModeCorners;        break;      case 3:        imageModeConvert = imageModeCenter;        break;      default:        throw "Invalid imageMode";      }    };    p.ellipseMode = function(aEllipseMode) {      curEllipseMode = aEllipseMode    };    p.arc = function(x, y, width, height, start, stop) {      if (width <= 0 || stop < start) return;      if (curEllipseMode === 1) {        width = width - x;        height = height - y      } else if (curEllipseMode === 2) {        x = x - width;        y = y - height;        width = width * 2;        height = height * 2      } else if (curEllipseMode === 3) {        x = x - width / 2;        y = y - height / 2      }      while (start < 0) {        start += 6.283185307179586;        stop += 6.283185307179586      }      if (stop - start > 6.283185307179586) {        start = 0;        stop = 6.283185307179586      }      var hr = width / 2,        vr = height / 2,        centerX = x + hr,        centerY = y + vr,        startLUT = 0 | 0.5 + start * p.RAD_TO_DEG * 2,        stopLUT = 0 | 0.5 + stop * p.RAD_TO_DEG * 2,        i, j;      if (doFill) {        var savedStroke = doStroke;        doStroke = false;        p.beginShape();        p.vertex(centerX, centerY);        for (i = startLUT; i <= stopLUT; i++) {          j = i % 720;          p.vertex(centerX + cosLUT[j] * hr, centerY + sinLUT[j] * vr)        }        p.endShape(2);        doStroke = savedStroke      }      if (doStroke) {        var savedFill = doFill;        doFill = false;        p.beginShape();        for (i = startLUT; i <= stopLUT; i++) {          j = i % 720;          p.vertex(centerX + cosLUT[j] * hr, centerY + sinLUT[j] * vr)        }        p.endShape();        doFill = savedFill      }    };    Drawing2D.prototype.line = function(x1, y1, x2, y2) {      if (!doStroke) return;      x1 = Math.round(x1);      x2 = Math.round(x2);      y1 = Math.round(y1);      y2 = Math.round(y2);      if (x1 === x2 && y1 === y2) {        p.point(x1, y1);        return      }      var swap = undef,        lineCap = undef,        drawCrisp = true,        currentModelView = modelView.array(),        identityMatrix = [1, 0, 0, 0, 1, 0];      for (var i = 0; i < 6 && drawCrisp; i++) drawCrisp = currentModelView[i] === identityMatrix[i];      if (drawCrisp) {        if (x1 === x2) {          if (y1 > y2) {            swap = y1;            y1 = y2;            y2 = swap          }          y2++;          if (lineWidth % 2 === 1) curContext.translate(0.5, 0)        } else if (y1 === y2) {          if (x1 > x2) {            swap = x1;            x1 = x2;            x2 = swap          }          x2++;          if (lineWidth % 2 === 1) curContext.translate(0, 0.5)        }        if (lineWidth === 1) {          lineCap = curContext.lineCap;          curContext.lineCap = "butt"        }      }      curContext.beginPath();      curContext.moveTo(x1 || 0, y1 || 0);      curContext.lineTo(x2 || 0, y2 || 0);      executeContextStroke();      if (drawCrisp) {        if (x1 === x2 && lineWidth % 2 === 1) curContext.translate(-0.5, 0);        else if (y1 === y2 && lineWidth % 2 === 1) curContext.translate(0, -0.5);        if (lineWidth === 1) curContext.lineCap = lineCap      }    };    Drawing3D.prototype.line = function(x1, y1, z1, x2, y2, z2) {      if (y2 === undef || z2 === undef) {        z2 = 0;        y2 = x2;        x2 = z1;        z1 = 0      }      if (x1 === x2 && y1 === y2 && z1 === z2) {        p.point(x1, y1, z1);        return      }      var lineVerts = [x1, y1, z1, x2, y2, z2];      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      if (lineWidth > 0 && doStroke) {        curContext.useProgram(programObject2D);        uniformMatrix("uModel2d", programObject2D, "uModel", false, [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]);        uniformMatrix("uView2d", programObject2D, "uView", false, view.array());        uniformf("uColor2d", programObject2D, "uColor", strokeStyle);        uniformi("uIsDrawingText", programObject2D, "uIsDrawingText", false);        vertexAttribPointer("aVertex2d", programObject2D, "aVertex", 3, lineBuffer);        disableVertexAttribPointer("aTextureCoord2d", programObject2D, "aTextureCoord");        curContext.bufferData(curContext.ARRAY_BUFFER, new Float32Array(lineVerts), curContext.STREAM_DRAW);        curContext.drawArrays(curContext.LINES, 0, 2)      }    };    Drawing2D.prototype.bezier = function() {      if (arguments.length !== 8) throw "You must use 8 parameters for bezier() in 2D mode";      p.beginShape();      p.vertex(arguments[0], arguments[1]);      p.bezierVertex(arguments[2], arguments[3], arguments[4], arguments[5], arguments[6], arguments[7]);      p.endShape()    };    Drawing3D.prototype.bezier = function() {      if (arguments.length !== 12) throw "You must use 12 parameters for bezier() in 3D mode";      p.beginShape();      p.vertex(arguments[0], arguments[1], arguments[2]);      p.bezierVertex(arguments[3], arguments[4], arguments[5], arguments[6], arguments[7], arguments[8], arguments[9], arguments[10], arguments[11]);      p.endShape()    };    p.bezierDetail = function(detail) {      bezDetail = detail    };    p.bezierPoint = function(a, b, c, d, t) {      return (1 - t) * (1 - t) * (1 - t) * a + 3 * (1 - t) * (1 - t) * t * b + 3 * (1 - t) * t * t * c + t * t * t * d    };    p.bezierTangent = function(a, b, c, d, t) {      return 3 * t * t * (-a + 3 * b - 3 * c + d) + 6 * t * (a - 2 * b + c) + 3 * (-a + b)    };    p.curvePoint = function(a, b, c, d, t) {      return 0.5 * (2 * b + (-a + c) * t + (2 * a - 5 * b + 4 * c - d) * t * t + (-a + 3 * b - 3 * c + d) * t * t * t)    };    p.curveTangent = function(a, b, c, d, t) {      return 0.5 * (-a + c + 2 * (2 * a - 5 * b + 4 * c - d) * t + 3 * (-a + 3 * b - 3 * c + d) * t * t)    };    p.triangle = function(x1, y1, x2, y2, x3, y3) {      p.beginShape(9);      p.vertex(x1, y1, 0);      p.vertex(x2, y2, 0);      p.vertex(x3, y3, 0);      p.endShape()    };    p.quad = function(x1, y1, x2, y2, x3, y3, x4, y4) {      p.beginShape(16);      p.vertex(x1, y1, 0);      p.vertex(x2, y2, 0);      p.vertex(x3, y3, 0);      p.vertex(x4, y4, 0);      p.endShape()    };    var roundedRect$2d = function(x, y, width, height, tl, tr, br, bl) {      if (bl === undef) {        tr = tl;        br = tl;        bl = tl      }      var halfWidth = width / 2,        halfHeight = height / 2;      if (tl > halfWidth || tl > halfHeight) tl = Math.min(halfWidth, halfHeight);      if (tr > halfWidth || tr > halfHeight) tr = Math.min(halfWidth, halfHeight);      if (br > halfWidth || br > halfHeight) br = Math.min(halfWidth, halfHeight);      if (bl > halfWidth || bl > halfHeight) bl = Math.min(halfWidth, halfHeight);      if (!doFill || doStroke) curContext.translate(0.5, 0.5);      curContext.beginPath();      curContext.moveTo(x + tl, y);      curContext.lineTo(x + width - tr, y);      curContext.quadraticCurveTo(x + width, y, x + width, y + tr);      curContext.lineTo(x + width, y + height - br);      curContext.quadraticCurveTo(x + width, y + height, x + width - br, y + height);      curContext.lineTo(x + bl, y + height);      curContext.quadraticCurveTo(x, y + height, x, y + height - bl);      curContext.lineTo(x, y + tl);      curContext.quadraticCurveTo(x, y, x + tl, y);      if (!doFill || doStroke) curContext.translate(-0.5, -0.5);      executeContextFill();      executeContextStroke()    };    Drawing2D.prototype.rect = function(x, y, width, height, tl, tr, br, bl) {      if (!width && !height) return;      if (curRectMode === 1) {        width -= x;        height -= y      } else if (curRectMode === 2) {        width *= 2;        height *= 2;        x -= width / 2;        y -= height / 2      } else if (curRectMode === 3) {        x -= width / 2;        y -= height / 2      }      x = Math.round(x);      y = Math.round(y);      width = Math.round(width);      height = Math.round(height);      if (tl !== undef) {        roundedRect$2d(x, y, width, height, tl, tr, br, bl);        return      }      if (doStroke && lineWidth % 2 === 1) curContext.translate(0.5, 0.5);      curContext.beginPath();      curContext.rect(x, y, width, height);      executeContextFill();      executeContextStroke();      if (doStroke && lineWidth % 2 === 1) curContext.translate(-0.5, -0.5)    };    Drawing3D.prototype.rect = function(x, y, width, height, tl, tr, br, bl) {      if (tl !== undef) throw "rect() with rounded corners is not supported in 3D mode";      if (curRectMode === 1) {        width -= x;        height -= y      } else if (curRectMode === 2) {        width *= 2;        height *= 2;        x -= width / 2;        y -= height / 2      } else if (curRectMode === 3) {        x -= width / 2;        y -= height / 2      }      var model = new PMatrix3D;      model.translate(x, y, 0);      model.scale(width, height, 1);      model.transpose();      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      if (lineWidth > 0 && doStroke) {        curContext.useProgram(programObject2D);        uniformMatrix("uModel2d", programObject2D, "uModel", false, model.array());        uniformMatrix("uView2d", programObject2D, "uView", false, view.array());        uniformf("uColor2d", programObject2D, "uColor", strokeStyle);        uniformi("uIsDrawingText2d", programObject2D, "uIsDrawingText", false);        vertexAttribPointer("aVertex2d", programObject2D, "aVertex", 3, rectBuffer);        disableVertexAttribPointer("aTextureCoord2d", programObject2D, "aTextureCoord");        curContext.drawArrays(curContext.LINE_LOOP, 0, rectVerts.length / 3)      }      if (doFill) {        curContext.useProgram(programObject3D);        uniformMatrix("uModel3d", programObject3D, "uModel", false, model.array());        uniformMatrix("uView3d", programObject3D, "uView", false, view.array());        curContext.enable(curContext.POLYGON_OFFSET_FILL);        curContext.polygonOffset(1, 1);        uniformf("color3d", programObject3D, "uColor", fillStyle);        if (lightCount > 0) {          var v = new PMatrix3D;          v.set(view);          var m = new PMatrix3D;          m.set(model);          v.mult(m);          var normalMatrix = new PMatrix3D;          normalMatrix.set(v);          normalMatrix.invert();          normalMatrix.transpose();          uniformMatrix("uNormalTransform3d", programObject3D, "uNormalTransform", false, normalMatrix.array());          vertexAttribPointer("aNormal3d", programObject3D, "aNormal", 3, rectNormBuffer)        } else disableVertexAttribPointer("normal3d", programObject3D, "aNormal");        vertexAttribPointer("vertex3d", programObject3D, "aVertex", 3, rectBuffer);        curContext.drawArrays(curContext.TRIANGLE_FAN, 0, rectVerts.length / 3);        curContext.disable(curContext.POLYGON_OFFSET_FILL)      }    };    Drawing2D.prototype.ellipse = function(x, y, width, height) {      x = x || 0;      y = y || 0;      if (width <= 0 && height <= 0) return;      if (curEllipseMode === 2) {        width *= 2;        height *= 2      } else if (curEllipseMode === 1) {        width = width - x;        height = height - y;        x += width / 2;        y += height / 2      } else if (curEllipseMode === 0) {        x += width / 2;        y += height / 2      }      if (width === height) {        curContext.beginPath();        curContext.arc(x, y, width / 2, 0, 6.283185307179586, false);        executeContextFill();        executeContextStroke()      } else {        var w = width / 2,          h = height / 2,          C = 0.5522847498307933,          c_x = C * w,          c_y = C * h;        p.beginShape();        p.vertex(x + w, y);        p.bezierVertex(x + w, y - c_y, x + c_x, y - h, x, y - h);        p.bezierVertex(x - c_x, y - h, x - w, y - c_y, x - w, y);        p.bezierVertex(x - w, y + c_y, x - c_x, y + h, x, y + h);        p.bezierVertex(x + c_x, y + h, x + w, y + c_y, x + w, y);        p.endShape()      }    };    Drawing3D.prototype.ellipse = function(x, y, width, height) {      x = x || 0;      y = y || 0;      if (width <= 0 && height <= 0) return;      if (curEllipseMode === 2) {        width *= 2;        height *= 2      } else if (curEllipseMode === 1) {        width = width - x;        height = height - y;        x += width / 2;        y += height / 2      } else if (curEllipseMode === 0) {        x += width / 2;        y += height / 2      }      var w = width / 2,        h = height / 2,        C = 0.5522847498307933,        c_x = C * w,        c_y = C * h;      p.beginShape();      p.vertex(x + w, y);      p.bezierVertex(x + w, y - c_y, 0, x + c_x, y - h, 0, x, y - h, 0);      p.bezierVertex(x - c_x, y - h, 0, x - w, y - c_y, 0, x - w, y, 0);      p.bezierVertex(x - w, y + c_y, 0, x - c_x, y + h, 0, x, y + h, 0);      p.bezierVertex(x + c_x, y + h, 0, x + w, y + c_y, 0, x + w, y, 0);      p.endShape();      if (doFill) {        var xAv = 0,          yAv = 0,          i, j;        for (i = 0; i < vertArray.length; i++) {          xAv += vertArray[i][0];          yAv += vertArray[i][1]        }        xAv /= vertArray.length;        yAv /= vertArray.length;        var vert = [],          fillVertArray = [],          colorVertArray = [];        vert[0] = xAv;        vert[1] = yAv;        vert[2] = 0;        vert[3] = 0;        vert[4] = 0;        vert[5] = fillStyle[0];        vert[6] = fillStyle[1];        vert[7] = fillStyle[2];        vert[8] = fillStyle[3];        vert[9] = strokeStyle[0];        vert[10] = strokeStyle[1];        vert[11] = strokeStyle[2];        vert[12] = strokeStyle[3];        vert[13] = normalX;        vert[14] = normalY;        vert[15] = normalZ;        vertArray.unshift(vert);        for (i = 0; i < vertArray.length; i++) {          for (j = 0; j < 3; j++) fillVertArray.push(vertArray[i][j]);          for (j = 5; j < 9; j++) colorVertArray.push(vertArray[i][j])        }        fill3D(fillVertArray, "TRIANGLE_FAN", colorVertArray)      }    };    p.normal = function(nx, ny, nz) {      if (arguments.length !== 3 || !(typeof nx === "number" && typeof ny === "number" && typeof nz === "number")) throw "normal() requires three numeric arguments.";      normalX = nx;      normalY = ny;      normalZ = nz;      if (curShape !== 0) if (normalMode === 0) normalMode = 1;      else if (normalMode === 1) normalMode = 2    };    p.save = function(file, img) {      if (img !== undef) return window.open(img.toDataURL(), "_blank");      return window.open(p.externals.canvas.toDataURL(), "_blank")    };    var saveNumber = 0;    p.saveFrame = function(file) {      if (file === undef) file = "screen-####.png";      var frameFilename = file.replace(/#+/, function(all) {        var s = "" + saveNumber++;        while (s.length < all.length) s = "0" + s;        return s      });      p.save(frameFilename)    };    var utilityContext2d = document.createElement("canvas").getContext("2d");    var canvasDataCache = [undef, undef, undef];    function getCanvasData(obj, w, h) {      var canvasData = canvasDataCache.shift();      if (canvasData === undef) {        canvasData = {};        canvasData.canvas = document.createElement("canvas");        canvasData.context = canvasData.canvas.getContext("2d")      }      canvasDataCache.push(canvasData);      var canvas = canvasData.canvas,        context = canvasData.context,        width = w || obj.width,        height = h || obj.height;      canvas.width = width;      canvas.height = height;      if (!obj) context.clearRect(0, 0, width, height);      else if ("data" in obj) context.putImageData(obj, 0, 0);      else {        context.clearRect(0, 0, width, height);        context.drawImage(obj, 0, 0, width, height)      }      return canvasData    }    function buildPixelsObject(pImage) {      return {        getLength: function(aImg) {          return function() {            if (aImg.isRemote) throw "Image is loaded remotely. Cannot get length.";            else return aImg.imageData.data.length ? aImg.imageData.data.length / 4 : 0          }        }(pImage),        getPixel: function(aImg) {          return function(i) {            var offset = i * 4,              data = aImg.imageData.data;            if (aImg.isRemote) throw "Image is loaded remotely. Cannot get pixels.";            return (data[offset + 3] & 255) << 24 | (data[offset] & 255) << 16 | (data[offset + 1] & 255) << 8 | data[offset + 2] & 255          }        }(pImage),        setPixel: function(aImg) {          return function(i, c) {            var offset = i * 4,              data = aImg.imageData.data;            if (aImg.isRemote) throw "Image is loaded remotely. Cannot set pixel.";            data[offset + 0] = (c >> 16) & 255;            data[offset + 1] = (c >> 8) & 255;            data[offset + 2] = c & 255;            data[offset + 3] = (c >> 24) & 255;            aImg.__isDirty = true          }        }(pImage),        toArray: function(aImg) {          return function() {            var arr = [],              data = aImg.imageData.data,              length = aImg.width * aImg.height;            if (aImg.isRemote) throw "Image is loaded remotely. Cannot get pixels.";            for (var i = 0, offset = 0; i < length; i++, offset += 4) arr.push((data[offset + 3] & 255) << 24 | (data[offset] & 255) << 16 | (data[offset + 1] & 255) << 8 | data[offset + 2] & 255);            return arr          }        }(pImage),        set: function(aImg) {          return function(arr) {            var offset, data, c;            if (this.isRemote) throw "Image is loaded remotely. Cannot set pixels.";            data = aImg.imageData.data;            for (var i = 0, aL = arr.length; i < aL; i++) {              c = arr[i];              offset = i * 4;              data[offset + 0] = (c >> 16) & 255;              data[offset + 1] = (c >> 8) & 255;              data[offset + 2] = c & 255;              data[offset + 3] = (c >> 24) & 255            }            aImg.__isDirty = true          }        }(pImage)      }    }    var PImage = function(aWidth, aHeight, aFormat) {      this.__isDirty = false;      if (aWidth instanceof HTMLImageElement) this.fromHTMLImageData(aWidth);      else if (aHeight || aFormat) {        this.width = aWidth || 1;        this.height = aHeight || 1;        var canvas = this.sourceImg = document.createElement("canvas");        canvas.width = this.width;        canvas.height = this.height;        var imageData = this.imageData = canvas.getContext("2d").createImageData(this.width, this.height);        this.format = aFormat === 2 || aFormat === 4 ? aFormat : 1;        if (this.format === 1) for (var i = 3, data = this.imageData.data, len = data.length; i < len; i += 4) data[i] = 255;        this.__isDirty = true;        this.updatePixels()      } else {        this.width = 0;        this.height = 0;        this.imageData = utilityContext2d.createImageData(1, 1);        this.format = 2      }      this.pixels = buildPixelsObject(this)    };    PImage.prototype = {      __isPImage: true,      updatePixels: function() {        var canvas = this.sourceImg;        if (canvas && canvas instanceof HTMLCanvasElement && this.__isDirty) canvas.getContext("2d").putImageData(this.imageData, 0, 0);        this.__isDirty = false      },      fromHTMLImageData: function(htmlImg) {        var canvasData = getCanvasData(htmlImg);        try {          var imageData = canvasData.context.getImageData(0, 0, htmlImg.width, htmlImg.height);          this.fromImageData(imageData)        } catch(e) {          if (htmlImg.width && htmlImg.height) {            this.isRemote = true;            this.width = htmlImg.width;            this.height = htmlImg.height          }        }        this.sourceImg = htmlImg      },      "get": function(x, y, w, h) {        if (!arguments.length) return p.get(this);        if (arguments.length === 2) return p.get(x, y, this);        if (arguments.length === 4) return p.get(x, y, w, h, this)      },      "set": function(x, y, c) {        p.set(x, y, c, this);        this.__isDirty = true      },      blend: function(srcImg, x, y, width, height, dx, dy, dwidth, dheight, MODE) {        if (arguments.length === 9) p.blend(this, srcImg, x, y, width, height, dx, dy, dwidth, dheight, this);        else if (arguments.length === 10) p.blend(srcImg, x, y, width, height, dx, dy, dwidth, dheight, MODE, this);        delete this.sourceImg      },      copy: function(srcImg, sx, sy, swidth, sheight, dx, dy, dwidth, dheight) {        if (arguments.length === 8) p.blend(this, srcImg, sx, sy, swidth, sheight, dx, dy, dwidth, 0, this);        else if (arguments.length === 9) p.blend(srcImg, sx, sy, swidth, sheight, dx, dy, dwidth, dheight, 0, this);        delete this.sourceImg      },      filter: function(mode, param) {        if (arguments.length === 2) p.filter(mode, param, this);        else if (arguments.length === 1) p.filter(mode, null, this);        delete this.sourceImg      },      save: function(file) {        p.save(file, this)      },      resize: function(w, h) {        if (this.isRemote) throw "Image is loaded remotely. Cannot resize.";        if (this.width !== 0 || this.height !== 0) {          if (w === 0 && h !== 0) w = Math.floor(this.width / this.height * h);          else if (h === 0 && w !== 0) h = Math.floor(this.height / this.width * w);          var canvas = getCanvasData(this.imageData).canvas;          var imageData = getCanvasData(canvas, w, h).context.getImageData(0, 0, w, h);          this.fromImageData(imageData)        }      },      mask: function(mask) {        var obj = this.toImageData(),          i, size;        if (mask instanceof PImage || mask.__isPImage) if (mask.width === this.width && mask.height === this.height) {          mask = mask.toImageData();          for (i = 2, size = this.width * this.height * 4; i < size; i += 4) obj.data[i + 1] = mask.data[i]        } else throw "mask must have the same dimensions as PImage.";        else if (mask instanceof        Array) if (this.width * this.height === mask.length) for (i = 0, size = mask.length; i < size; ++i) obj.data[i * 4 + 3] = mask[i];        else throw "mask array must be the same length as PImage pixels array.";        this.fromImageData(obj)      },      loadPixels: nop,      toImageData: function() {        if (this.isRemote) return this.sourceImg;        if (!this.__isDirty) return this.imageData;        var canvasData = getCanvasData(this.sourceImg);        return canvasData.context.getImageData(0, 0, this.width, this.height)      },      toDataURL: function() {        if (this.isRemote) throw "Image is loaded remotely. Cannot create dataURI.";        var canvasData = getCanvasData(this.imageData);        return canvasData.canvas.toDataURL()      },      fromImageData: function(canvasImg) {        var w = canvasImg.width,          h = canvasImg.height,          canvas = document.createElement("canvas"),          ctx = canvas.getContext("2d");        this.width = canvas.width = w;        this.height = canvas.height = h;        ctx.putImageData(canvasImg, 0, 0);        this.format = 2;        this.imageData = canvasImg;        this.sourceImg = canvas      }    };    p.PImage = PImage;    p.createImage = function(w, h, mode) {      return new PImage(w, h, mode)    };    p.loadImage = function(file, type, callback) {      if (type) file = file + "." + type;      var pimg;      if (curSketch.imageCache.images[file]) {        pimg = new PImage(curSketch.imageCache.images[file]);        pimg.loaded = true;        return pimg      }      pimg = new PImage;      var img = document.createElement("img");      pimg.sourceImg = img;      img.onload = function(aImage, aPImage, aCallback) {        var image = aImage;        var pimg = aPImage;        var callback = aCallback;        return function() {          pimg.fromHTMLImageData(image);          pimg.loaded = true;          if (callback) callback()        }      }(img, pimg, callback);      img.src = file;      return pimg    };    p.requestImage = p.loadImage;    function get$2(x, y) {      var data;      if (x >= p.width || x < 0 || y < 0 || y >= p.height) return 0;      if (isContextReplaced) {        var offset = ((0 | x) + p.width * (0 | y)) * 4;        data = p.imageData.data;        return (data[offset + 3] & 255) << 24 | (data[offset] & 255) << 16 | (data[offset + 1] & 255) << 8 | data[offset + 2] & 255      }      data = p.toImageData(0 | x, 0 | y, 1, 1).data;      return (data[3] & 255) << 24 | (data[0] & 255) << 16 | (data[1] & 255) << 8 | data[2] & 255    }    function get$3(x, y, img) {      if (img.isRemote) throw "Image is loaded remotely. Cannot get x,y.";      var offset = y * img.width * 4 + x * 4,        data = img.imageData.data;      return (data[offset + 3] & 255) << 24 | (data[offset] & 255) << 16 | (data[offset + 1] & 255) << 8 | data[offset + 2] & 255    }    function get$4(x, y, w, h) {      var c = new PImage(w, h, 2);      c.fromImageData(p.toImageData(x, y, w, h));      return c    }    function get$5(x, y, w, h, img) {      if (img.isRemote) throw "Image is loaded remotely. Cannot get x,y,w,h.";      var c = new PImage(w, h, 2),        cData = c.imageData.data,        imgWidth = img.width,        imgHeight = img.height,        imgData = img.imageData.data;      var startRow = Math.max(0, -y),        startColumn = Math.max(0, -x),        stopRow = Math.min(h, imgHeight - y),        stopColumn = Math.min(w, imgWidth - x);      for (var i = startRow; i < stopRow; ++i) {        var sourceOffset = ((y + i) * imgWidth + (x + startColumn)) * 4;        var targetOffset = (i * w + startColumn) * 4;        for (var j = startColumn; j < stopColumn; ++j) {          cData[targetOffset++] = imgData[sourceOffset++];          cData[targetOffset++] = imgData[sourceOffset++];          cData[targetOffset++] = imgData[sourceOffset++];          cData[targetOffset++] = imgData[sourceOffset++]        }      }      c.__isDirty = true;      return c    }    p.get = function(x, y, w, h, img) {      if (img !== undefined) return get$5(x, y, w, h, img);      if (h !== undefined) return get$4(x, y, w, h);      if (w !== undefined) return get$3(x, y, w);      if (y !== undefined) return get$2(x, y);      if (x !== undefined) return get$5(0, 0, x.width, x.height, x);      return get$4(0, 0, p.width, p.height)    };    p.createGraphics = function(w, h, render) {      var pg = new Processing;      pg.size(w, h, render);      pg.background(0, 0);      return pg    };    function resetContext() {      if (isContextReplaced) {        curContext = originalContext;        isContextReplaced = false;        p.updatePixels()      }    }    function SetPixelContextWrapper() {      function wrapFunction(newContext, name) {        function wrapper() {          resetContext();          curContext[name].apply(curContext, arguments)        }        newContext[name] = wrapper      }      function wrapProperty(newContext, name) {        function getter() {          resetContext();          return curContext[name]        }        function setter(value) {          resetContext();          curContext[name] = value        }        p.defineProperty(newContext, name, {          get: getter,          set: setter        })      }      for (var n in curContext) if (typeof curContext[n] === "function") wrapFunction(this, n);      else wrapProperty(this, n)    }    function replaceContext() {      if (isContextReplaced) return;      p.loadPixels();      if (proxyContext === null) {        originalContext = curContext;        proxyContext = new SetPixelContextWrapper      }      isContextReplaced = true;      curContext = proxyContext;      setPixelsCached = 0    }    function set$3(x, y, c) {      if (x < p.width && x >= 0 && y >= 0 && y < p.height) {        replaceContext();        p.pixels.setPixel((0 | x) + p.width * (0 | y), c);        if (++setPixelsCached > maxPixelsCached) resetContext()      }    }    function set$4(x, y, obj, img) {      if (img.isRemote) throw "Image is loaded remotely. Cannot set x,y.";      var c = p.color.toArray(obj);      var offset = y * img.width * 4 + x * 4;      var data = img.imageData.data;      data[offset] = c[0];      data[offset + 1] = c[1];      data[offset + 2] = c[2];      data[offset + 3] = c[3]    }    p.set = function(x, y, obj, img) {      var color, oldFill;      if (arguments.length === 3) if (typeof obj === "number") set$3(x, y, obj);      else {        if (obj instanceof PImage || obj.__isPImage) p.image(obj, x, y)      } else if (arguments.length === 4) set$4(x, y, obj, img)    };    p.imageData = {};    p.pixels = {      getLength: function() {        return p.imageData.data.length ? p.imageData.data.length / 4 : 0      },      getPixel: function(i) {        var offset = i * 4,          data = p.imageData.data;        return data[offset + 3] << 24 & 4278190080 | data[offset + 0] << 16 & 16711680 | data[offset + 1] << 8 & 65280 | data[offset + 2] & 255      },      setPixel: function(i, c) {        var offset = i * 4,          data = p.imageData.data;        data[offset + 0] = (c & 16711680) >>> 16;        data[offset + 1] = (c & 65280) >>> 8;        data[offset + 2] = c & 255;        data[offset + 3] = (c & 4278190080) >>> 24      },      toArray: function() {        var arr = [],          length = p.imageData.width * p.imageData.height,          data = p.imageData.data;        for (var i = 0, offset = 0; i < length; i++, offset += 4) arr.push(data[offset + 3] << 24 & 4278190080 | data[offset + 0] << 16 & 16711680 | data[offset + 1] << 8 & 65280 | data[offset + 2] & 255);        return arr      },      set: function(arr) {        for (var i = 0, aL = arr.length; i < aL; i++) this.setPixel(i, arr[i])      }    };    p.loadPixels = function() {      p.imageData = drawing.$ensureContext().getImageData(0, 0, p.width, p.height)    };    p.updatePixels = function() {      if (p.imageData) drawing.$ensureContext().putImageData(p.imageData, 0, 0)    };    p.hint = function(which) {      var curContext = drawing.$ensureContext();      if (which === 4) {        curContext.disable(curContext.DEPTH_TEST);        curContext.depthMask(false);        curContext.clear(curContext.DEPTH_BUFFER_BIT)      } else if (which === -4) {        curContext.enable(curContext.DEPTH_TEST);        curContext.depthMask(true)      } else if (which === -1 || which === 2) renderSmooth = true;      else if (which === 1) renderSmooth = false    };    var backgroundHelper = function(arg1, arg2, arg3, arg4) {      var obj;      if (arg1 instanceof PImage || arg1.__isPImage) {        obj = arg1;        if (!obj.loaded) throw "Error using image in background(): PImage not loaded.";        if (obj.width !== p.width || obj.height !== p.height) throw "Background image must be the same dimensions as the canvas.";      } else obj = p.color(arg1, arg2, arg3, arg4);      backgroundObj = obj    };    Drawing2D.prototype.background = function(arg1, arg2, arg3, arg4) {      if (arg1 !== undef) backgroundHelper(arg1, arg2, arg3, arg4);      if (backgroundObj instanceof PImage || backgroundObj.__isPImage) {        saveContext();        curContext.setTransform(1, 0, 0, 1, 0, 0);        p.image(backgroundObj, 0, 0);        restoreContext()      } else {        saveContext();        curContext.setTransform(1, 0, 0, 1, 0, 0);        if (p.alpha(backgroundObj) !== colorModeA) curContext.clearRect(0, 0, p.width, p.height);        curContext.fillStyle = p.color.toString(backgroundObj);        curContext.fillRect(0, 0, p.width, p.height);        isFillDirty = true;        restoreContext()      }    };    Drawing3D.prototype.background = function(arg1, arg2, arg3, arg4) {      if (arguments.length > 0) backgroundHelper(arg1, arg2, arg3, arg4);      var c = p.color.toGLArray(backgroundObj);      curContext.clearColor(c[0], c[1], c[2], c[3]);      curContext.clear(curContext.COLOR_BUFFER_BIT | curContext.DEPTH_BUFFER_BIT)    };    Drawing2D.prototype.image = function(img, x, y, w, h) {      x = Math.round(x);      y = Math.round(y);      if (img.width > 0) {        var wid = w || img.width;        var hgt = h || img.height;        var bounds = imageModeConvert(x || 0, y || 0, w || img.width, h || img.height, arguments.length < 4);        var fastImage = !!img.sourceImg && curTint === null;        if (fastImage) {          var htmlElement = img.sourceImg;          if (img.__isDirty) img.updatePixels();          curContext.drawImage(htmlElement, 0, 0, htmlElement.width, htmlElement.height, bounds.x, bounds.y, bounds.w, bounds.h)        } else {          var obj = img.toImageData();          if (curTint !== null) {            curTint(obj);            img.__isDirty = true          }          curContext.drawImage(getCanvasData(obj).canvas, 0, 0, img.width, img.height, bounds.x, bounds.y, bounds.w, bounds.h)        }      }    };    Drawing3D.prototype.image = function(img, x, y, w, h) {      if (img.width > 0) {        x = Math.round(x);        y = Math.round(y);        w = w || img.width;        h = h || img.height;        p.beginShape(p.QUADS);        p.texture(img);        p.vertex(x, y, 0, 0, 0);        p.vertex(x, y + h, 0, 0, h);        p.vertex(x + w, y + h, 0, w, h);        p.vertex(x + w, y, 0, w, 0);        p.endShape()      }    };    p.tint = function(a1, a2, a3, a4) {      var tintColor = p.color(a1, a2, a3, a4);      var r = p.red(tintColor) / colorModeX;      var g = p.green(tintColor) / colorModeY;      var b = p.blue(tintColor) / colorModeZ;      var a = p.alpha(tintColor) / colorModeA;      curTint = function(obj) {        var data = obj.data,          length = 4 * obj.width * obj.height;        for (var i = 0; i < length;) {          data[i++] *= r;          data[i++] *= g;          data[i++] *= b;          data[i++] *= a        }      };      curTint3d = function(data) {        for (var i = 0; i < data.length;) {          data[i++] = r;          data[i++] = g;          data[i++] = b;          data[i++] = a        }      }    };    p.noTint = function() {      curTint = null;      curTint3d = null    };    p.copy = function(src, sx, sy, sw, sh, dx, dy, dw, dh) {      if (dh === undef) {        dh = dw;        dw = dy;        dy = dx;        dx = sh;        sh = sw;        sw = sy;        sy = sx;        sx = src;        src = p      }      p.blend(src, sx, sy, sw, sh, dx, dy, dw, dh, 0)    };    p.blend = function(src, sx, sy, sw, sh, dx, dy, dw, dh, mode, pimgdest) {      if (src.isRemote) throw "Image is loaded remotely. Cannot blend image.";      if (mode === undef) {        mode = dh;        dh = dw;        dw = dy;        dy = dx;        dx = sh;        sh = sw;        sw = sy;        sy = sx;        sx = src;        src = p      }      var sx2 = sx + sw,        sy2 = sy + sh,        dx2 = dx + dw,        dy2 = dy + dh,        dest = pimgdest || p;      if (pimgdest === undef || mode === undef) p.loadPixels();      src.loadPixels();      if (src === p && p.intersect(sx, sy, sx2, sy2, dx, dy, dx2, dy2)) p.blit_resize(p.get(sx, sy, sx2 - sx, sy2 - sy), 0, 0, sx2 - sx - 1, sy2 - sy - 1, dest.imageData.data, dest.width, dest.height, dx, dy, dx2, dy2, mode);      else p.blit_resize(src, sx, sy, sx2, sy2, dest.imageData.data, dest.width, dest.height, dx, dy, dx2, dy2, mode);      if (pimgdest === undef) p.updatePixels()    };    var buildBlurKernel = function(r) {      var radius = p.floor(r * 3.5),        i, radiusi;      radius = radius < 1 ? 1 : radius < 248 ? radius : 248;      if (p.shared.blurRadius !== radius) {        p.shared.blurRadius = radius;        p.shared.blurKernelSize = 1 + (p.shared.blurRadius << 1);        p.shared.blurKernel = new Float32Array(p.shared.blurKernelSize);        var sharedBlurKernal = p.shared.blurKernel;        var sharedBlurKernelSize = p.shared.blurKernelSize;        var sharedBlurRadius = p.shared.blurRadius;        for (i = 0; i < sharedBlurKernelSize; i++) sharedBlurKernal[i] = 0;        var radiusiSquared = (radius - 1) * (radius - 1);        for (i = 1; i < radius; i++) sharedBlurKernal[radius + i] = sharedBlurKernal[radiusi] = radiusiSquared;        sharedBlurKernal[radius] = radius * radius      }    };    var blurARGB = function(r, aImg) {      var sum, cr, cg, cb, ca, c, m;      var read, ri, ym, ymi, bk0;      var wh = aImg.pixels.getLength();      var r2 = new Float32Array(wh);      var g2 = new Float32Array(wh);      var b2 = new Float32Array(wh);      var a2 = new Float32Array(wh);      var yi = 0;      var x, y, i, offset;      buildBlurKernel(r);      var aImgHeight = aImg.height;      var aImgWidth = aImg.width;      var sharedBlurKernelSize = p.shared.blurKernelSize;      var sharedBlurRadius = p.shared.blurRadius;      var sharedBlurKernal = p.shared.blurKernel;      var pix = aImg.imageData.data;      for (y = 0; y < aImgHeight; y++) {        for (x = 0; x < aImgWidth; x++) {          cb = cg = cr = ca = sum = 0;          read = x - sharedBlurRadius;          if (read < 0) {            bk0 = -read;            read = 0          } else {            if (read >= aImgWidth) break;            bk0 = 0          }          for (i = bk0; i < sharedBlurKernelSize; i++) {            if (read >= aImgWidth) break;            offset = (read + yi) * 4;            m = sharedBlurKernal[i];            ca += m * pix[offset + 3];            cr += m * pix[offset];            cg += m * pix[offset + 1];            cb += m * pix[offset + 2];            sum += m;            read++          }          ri = yi + x;          a2[ri] = ca / sum;          r2[ri] = cr / sum;          g2[ri] = cg / sum;          b2[ri] = cb / sum        }        yi += aImgWidth      }      yi = 0;      ym = -sharedBlurRadius;      ymi = ym * aImgWidth;      for (y = 0; y < aImgHeight; y++) {        for (x = 0; x < aImgWidth; x++) {          cb = cg = cr = ca = sum = 0;          if (ym < 0) {            bk0 = ri = -ym;            read = x          } else {            if (ym >= aImgHeight) break;            bk0 = 0;            ri = ym;            read = x + ymi          }          for (i = bk0; i < sharedBlurKernelSize; i++) {            if (ri >= aImgHeight) break;            m = sharedBlurKernal[i];            ca += m * a2[read];            cr += m * r2[read];            cg += m * g2[read];            cb += m * b2[read];            sum += m;            ri++;            read += aImgWidth          }          offset = (x + yi) * 4;          pix[offset] = cr / sum;          pix[offset + 1] = cg / sum;          pix[offset + 2] = cb / sum;          pix[offset + 3] = ca / sum        }        yi += aImgWidth;        ymi += aImgWidth;        ym++      }    };    var dilate = function(isInverted, aImg) {      var currIdx = 0;      var maxIdx = aImg.pixels.getLength();      var out = new Int32Array(maxIdx);      var currRowIdx, maxRowIdx, colOrig, colOut, currLum;      var idxRight, idxLeft, idxUp, idxDown, colRight, colLeft, colUp, colDown, lumRight, lumLeft, lumUp, lumDown;      if (!isInverted) while (currIdx < maxIdx) {        currRowIdx = currIdx;        maxRowIdx = currIdx + aImg.width;        while (currIdx < maxRowIdx) {          colOrig = colOut = aImg.pixels.getPixel(currIdx);          idxLeft = currIdx - 1;          idxRight = currIdx + 1;          idxUp = currIdx - aImg.width;          idxDown = currIdx + aImg.width;          if (idxLeft < currRowIdx) idxLeft = currIdx;          if (idxRight >= maxRowIdx) idxRight = currIdx;          if (idxUp < 0) idxUp = 0;          if (idxDown >= maxIdx) idxDown = currIdx;          colUp = aImg.pixels.getPixel(idxUp);          colLeft = aImg.pixels.getPixel(idxLeft);          colDown = aImg.pixels.getPixel(idxDown);          colRight = aImg.pixels.getPixel(idxRight);          currLum = 77 * (colOrig >> 16 & 255) + 151 * (colOrig >> 8 & 255) + 28 * (colOrig & 255);          lumLeft = 77 * (colLeft >> 16 & 255) + 151 * (colLeft >> 8 & 255) + 28 * (colLeft & 255);          lumRight = 77 * (colRight >> 16 & 255) + 151 * (colRight >> 8 & 255) + 28 * (colRight & 255);          lumUp = 77 * (colUp >> 16 & 255) + 151 * (colUp >> 8 & 255) + 28 * (colUp & 255);          lumDown = 77 * (colDown >> 16 & 255) + 151 * (colDown >> 8 & 255) + 28 * (colDown & 255);          if (lumLeft > currLum) {            colOut = colLeft;            currLum = lumLeft          }          if (lumRight > currLum) {            colOut = colRight;            currLum = lumRight          }          if (lumUp > currLum) {            colOut = colUp;            currLum = lumUp          }          if (lumDown > currLum) {            colOut = colDown;            currLum = lumDown          }          out[currIdx++] = colOut        }      } else while (currIdx < maxIdx) {        currRowIdx = currIdx;        maxRowIdx = currIdx + aImg.width;        while (currIdx < maxRowIdx) {          colOrig = colOut = aImg.pixels.getPixel(currIdx);          idxLeft = currIdx - 1;          idxRight = currIdx + 1;          idxUp = currIdx - aImg.width;          idxDown = currIdx + aImg.width;          if (idxLeft < currRowIdx) idxLeft = currIdx;          if (idxRight >= maxRowIdx) idxRight = currIdx;          if (idxUp < 0) idxUp = 0;          if (idxDown >= maxIdx) idxDown = currIdx;          colUp = aImg.pixels.getPixel(idxUp);          colLeft = aImg.pixels.getPixel(idxLeft);          colDown = aImg.pixels.getPixel(idxDown);          colRight = aImg.pixels.getPixel(idxRight);          currLum = 77 * (colOrig >> 16 & 255) + 151 * (colOrig >> 8 & 255) + 28 * (colOrig & 255);          lumLeft = 77 * (colLeft >> 16 & 255) + 151 * (colLeft >> 8 & 255) + 28 * (colLeft & 255);          lumRight = 77 * (colRight >> 16 & 255) + 151 * (colRight >> 8 & 255) + 28 * (colRight & 255);          lumUp = 77 * (colUp >> 16 & 255) + 151 * (colUp >> 8 & 255) + 28 * (colUp & 255);          lumDown = 77 * (colDown >> 16 & 255) + 151 * (colDown >> 8 & 255) + 28 * (colDown & 255);          if (lumLeft < currLum) {            colOut = colLeft;            currLum = lumLeft          }          if (lumRight < currLum) {            colOut = colRight;            currLum = lumRight          }          if (lumUp < currLum) {            colOut = colUp;            currLum = lumUp          }          if (lumDown < currLum) {            colOut = colDown;            currLum = lumDown          }          out[currIdx++] = colOut        }      }      aImg.pixels.set(out)    };    p.filter = function(kind, param, aImg) {      var img, col, lum, i;      if (arguments.length === 3) {        aImg.loadPixels();        img = aImg      } else {        p.loadPixels();        img = p      }      if (param === undef) param = null;      if (img.isRemote) throw "Image is loaded remotely. Cannot filter image.";      var imglen = img.pixels.getLength();      switch (kind) {      case 11:        var radius = param || 1;        blurARGB(radius, img);        break;      case 12:        if (img.format === 4) {          for (i = 0; i < imglen; i++) {            col = 255 - img.pixels.getPixel(i);            img.pixels.setPixel(i, 4278190080 | col << 16 | col << 8 | col)          }          img.format = 1        } else for (i = 0; i < imglen; i++) {          col = img.pixels.getPixel(i);          lum = 77 * (col >> 16 & 255) + 151 * (col >> 8 & 255) + 28 * (col & 255) >> 8;          img.pixels.setPixel(i, col & 4278190080 | lum << 16 | lum << 8 | lum)        }        break;      case 13:        for (i = 0; i < imglen; i++) img.pixels.setPixel(i, img.pixels.getPixel(i) ^ 16777215);        break;      case 15:        if (param === null) throw "Use filter(POSTERIZE, int levels) instead of filter(POSTERIZE)";        var levels = p.floor(param);        if (levels < 2 || levels > 255) throw "Levels must be between 2 and 255 for filter(POSTERIZE, levels)";        var levels1 = levels - 1;        for (i = 0; i < imglen; i++) {          var rlevel = img.pixels.getPixel(i) >> 16 & 255;          var glevel = img.pixels.getPixel(i) >> 8 & 255;          var blevel = img.pixels.getPixel(i) & 255;          rlevel = (rlevel * levels >> 8) * 255 / levels1;          glevel = (glevel * levels >> 8) * 255 / levels1;          blevel = (blevel * levels >> 8) * 255 / levels1;          img.pixels.setPixel(i, 4278190080 & img.pixels.getPixel(i) | rlevel << 16 | glevel << 8 | blevel)        }        break;      case 14:        for (i = 0; i < imglen; i++) img.pixels.setPixel(i, img.pixels.getPixel(i) | 4278190080);        img.format = 1;        break;      case 16:        if (param === null) param = 0.5;        if (param < 0 || param > 1) throw "Level must be between 0 and 1 for filter(THRESHOLD, level)";        var thresh = p.floor(param * 255);        for (i = 0; i < imglen; i++) {          var max = p.max((img.pixels.getPixel(i) & 16711680) >> 16, p.max((img.pixels.getPixel(i) & 65280) >> 8, img.pixels.getPixel(i) & 255));          img.pixels.setPixel(i, img.pixels.getPixel(i) & 4278190080 | (max < thresh ? 0 : 16777215))        }        break;      case 17:        dilate(true, img);        break;      case 18:        dilate(false, img);        break      }      img.updatePixels()    };    p.shared = {      fracU: 0,      ifU: 0,      fracV: 0,      ifV: 0,      u1: 0,      u2: 0,      v1: 0,      v2: 0,      sX: 0,      sY: 0,      iw: 0,      iw1: 0,      ih1: 0,      ul: 0,      ll: 0,      ur: 0,      lr: 0,      cUL: 0,      cLL: 0,      cUR: 0,      cLR: 0,      srcXOffset: 0,      srcYOffset: 0,      r: 0,      g: 0,      b: 0,      a: 0,      srcBuffer: null,      blurRadius: 0,      blurKernelSize: 0,      blurKernel: null    };    p.intersect = function(sx1, sy1, sx2, sy2, dx1, dy1, dx2, dy2) {      var sw = sx2 - sx1 + 1;      var sh = sy2 - sy1 + 1;      var dw = dx2 - dx1 + 1;      var dh = dy2 - dy1 + 1;      if (dx1 < sx1) {        dw += dx1 - sx1;        if (dw > sw) dw = sw      } else {        var w = sw + sx1 - dx1;        if (dw > w) dw = w      }      if (dy1 < sy1) {        dh += dy1 - sy1;        if (dh > sh) dh = sh      } else {        var h = sh + sy1 - dy1;        if (dh > h) dh = h      }      return ! (dw <= 0 || dh <= 0)    };    var blendFuncs = {};    blendFuncs[1] = p.modes.blend;    blendFuncs[2] = p.modes.add;    blendFuncs[4] = p.modes.subtract;    blendFuncs[8] = p.modes.lightest;    blendFuncs[16] = p.modes.darkest;    blendFuncs[0] = p.modes.replace;    blendFuncs[32] = p.modes.difference;    blendFuncs[64] = p.modes.exclusion;    blendFuncs[128] = p.modes.multiply;    blendFuncs[256] = p.modes.screen;    blendFuncs[512] = p.modes.overlay;    blendFuncs[1024] = p.modes.hard_light;    blendFuncs[2048] = p.modes.soft_light;    blendFuncs[4096] = p.modes.dodge;    blendFuncs[8192] = p.modes.burn;    p.blit_resize = function(img, srcX1, srcY1, srcX2, srcY2, destPixels, screenW, screenH, destX1, destY1, destX2, destY2, mode) {      var x, y;      if (srcX1 < 0) srcX1 = 0;      if (srcY1 < 0) srcY1 = 0;      if (srcX2 >= img.width) srcX2 = img.width - 1;      if (srcY2 >= img.height) srcY2 = img.height - 1;      var srcW = srcX2 - srcX1;      var srcH = srcY2 - srcY1;      var destW = destX2 - destX1;      var destH = destY2 - destY1;      if (destW <= 0 || destH <= 0 || srcW <= 0 || srcH <= 0 || destX1 >= screenW || destY1 >= screenH || srcX1 >= img.width || srcY1 >= img.height) return;      var dx = Math.floor(srcW / destW * 32768);      var dy = Math.floor(srcH / destH * 32768);      var pshared = p.shared;      pshared.srcXOffset = Math.floor(destX1 < 0 ? -destX1 * dx : srcX1 * 32768);      pshared.srcYOffset = Math.floor(destY1 < 0 ? -destY1 * dy : srcY1 * 32768);      if (destX1 < 0) {        destW += destX1;        destX1 = 0      }      if (destY1 < 0) {        destH += destY1;        destY1 = 0      }      destW = Math.min(destW, screenW - destX1);      destH = Math.min(destH, screenH - destY1);      var destOffset = destY1 * screenW + destX1;      var destColor;      pshared.srcBuffer = img.imageData.data;      pshared.iw = img.width;      pshared.iw1 = img.width - 1;      pshared.ih1 = img.height - 1;      var filterBilinear = p.filter_bilinear,        filterNewScanline = p.filter_new_scanline,        blendFunc = blendFuncs[mode],        blendedColor, idx, cULoffset, cURoffset, cLLoffset, cLRoffset, ALPHA_MASK = 4278190080,        RED_MASK = 16711680,        GREEN_MASK = 65280,        BLUE_MASK = 255,        PREC_MAXVAL = 32767,        PRECISIONB = 15,        PREC_RED_SHIFT = 1,        PREC_ALPHA_SHIFT = 9,        srcBuffer = pshared.srcBuffer,        min = Math.min;      for (y = 0; y < destH; y++) {        pshared.sX = pshared.srcXOffset;        pshared.fracV = pshared.srcYOffset & PREC_MAXVAL;        pshared.ifV = PREC_MAXVAL - pshared.fracV;        pshared.v1 = (pshared.srcYOffset >> PRECISIONB) * pshared.iw;        pshared.v2 = min((pshared.srcYOffset >> PRECISIONB) + 1, pshared.ih1) * pshared.iw;        for (x = 0; x < destW; x++) {          idx = (destOffset + x) * 4;          destColor = destPixels[idx + 3] << 24 & ALPHA_MASK | destPixels[idx] << 16 & RED_MASK | destPixels[idx + 1] << 8 & GREEN_MASK | destPixels[idx + 2] & BLUE_MASK;          pshared.fracU = pshared.sX & PREC_MAXVAL;          pshared.ifU = PREC_MAXVAL - pshared.fracU;          pshared.ul = pshared.ifU * pshared.ifV >> PRECISIONB;          pshared.ll = pshared.ifU * pshared.fracV >> PRECISIONB;          pshared.ur = pshared.fracU * pshared.ifV >> PRECISIONB;          pshared.lr = pshared.fracU * pshared.fracV >> PRECISIONB;          pshared.u1 = pshared.sX >> PRECISIONB;          pshared.u2 = min(pshared.u1 + 1, pshared.iw1);          cULoffset = (pshared.v1 + pshared.u1) * 4;          cURoffset = (pshared.v1 + pshared.u2) * 4;          cLLoffset = (pshared.v2 + pshared.u1) * 4;          cLRoffset = (pshared.v2 + pshared.u2) * 4;          pshared.cUL = srcBuffer[cULoffset + 3] << 24 & ALPHA_MASK | srcBuffer[cULoffset] << 16 & RED_MASK | srcBuffer[cULoffset + 1] << 8 & GREEN_MASK | srcBuffer[cULoffset + 2] & BLUE_MASK;          pshared.cUR = srcBuffer[cURoffset + 3] << 24 & ALPHA_MASK | srcBuffer[cURoffset] << 16 & RED_MASK | srcBuffer[cURoffset + 1] << 8 & GREEN_MASK | srcBuffer[cURoffset + 2] & BLUE_MASK;          pshared.cLL = srcBuffer[cLLoffset + 3] << 24 & ALPHA_MASK | srcBuffer[cLLoffset] << 16 & RED_MASK | srcBuffer[cLLoffset + 1] << 8 & GREEN_MASK | srcBuffer[cLLoffset + 2] & BLUE_MASK;          pshared.cLR = srcBuffer[cLRoffset + 3] << 24 & ALPHA_MASK | srcBuffer[cLRoffset] << 16 & RED_MASK | srcBuffer[cLRoffset + 1] << 8 & GREEN_MASK | srcBuffer[cLRoffset + 2] & BLUE_MASK;          pshared.r = pshared.ul * ((pshared.cUL & RED_MASK) >> 16) + pshared.ll * ((pshared.cLL & RED_MASK) >> 16) + pshared.ur * ((pshared.cUR & RED_MASK) >> 16) + pshared.lr * ((pshared.cLR & RED_MASK) >> 16) << PREC_RED_SHIFT & RED_MASK;          pshared.g = pshared.ul * (pshared.cUL & GREEN_MASK) + pshared.ll * (pshared.cLL & GREEN_MASK) + pshared.ur * (pshared.cUR & GREEN_MASK) + pshared.lr * (pshared.cLR & GREEN_MASK) >>> PRECISIONB & GREEN_MASK;          pshared.b = pshared.ul * (pshared.cUL & BLUE_MASK) + pshared.ll * (pshared.cLL & BLUE_MASK) + pshared.ur * (pshared.cUR & BLUE_MASK) + pshared.lr * (pshared.cLR & BLUE_MASK) >>> PRECISIONB;          pshared.a = pshared.ul * ((pshared.cUL & ALPHA_MASK) >>> 24) + pshared.ll * ((pshared.cLL & ALPHA_MASK) >>> 24) + pshared.ur * ((pshared.cUR & ALPHA_MASK) >>> 24) + pshared.lr * ((pshared.cLR & ALPHA_MASK) >>> 24) << PREC_ALPHA_SHIFT & ALPHA_MASK;          blendedColor = blendFunc(destColor, pshared.a | pshared.r | pshared.g | pshared.b);          destPixels[idx] = (blendedColor & RED_MASK) >>> 16;          destPixels[idx + 1] = (blendedColor & GREEN_MASK) >>> 8;          destPixels[idx + 2] = blendedColor & BLUE_MASK;          destPixels[idx + 3] = (blendedColor & ALPHA_MASK) >>> 24;          pshared.sX += dx        }        destOffset += screenW;        pshared.srcYOffset += dy      }    };    p.loadFont = function(name, size) {      if (name === undef) throw "font name required in loadFont.";      if (name.indexOf(".svg") === -1) {        if (size === undef) size = curTextFont.size;        return PFont.get(name, size)      }      var font = p.loadGlyphs(name);      return {        name: name,        css: "12px sans-serif",        glyph: true,        units_per_em: font.units_per_em,        horiz_adv_x: 1 / font.units_per_em * font.horiz_adv_x,        ascent: font.ascent,        descent: font.descent,        width: function(str) {          var width = 0;          var len = str.length;          for (var i = 0; i < len; i++) try {            width += parseFloat(p.glyphLook(p.glyphTable[name], str[i]).horiz_adv_x)          } catch(e) {            Processing.debug(e)          }          return width / p.glyphTable[name].units_per_em        }      }    };    p.createFont = function(name, size) {      return p.loadFont(name, size)    };    p.textFont = function(pfont, size) {      if (size !== undef) {        if (!pfont.glyph) pfont = PFont.get(pfont.name, size);        curTextSize = size      }      curTextFont = pfont;      curFontName = curTextFont.name;      curTextAscent = curTextFont.ascent;      curTextDescent = curTextFont.descent;      curTextLeading = curTextFont.leading;      var curContext = drawing.$ensureContext();      curContext.font = curTextFont.css    };    p.textSize = function(size) {      curTextFont = PFont.get(curFontName, size);      curTextSize = size;      curTextAscent = curTextFont.ascent;      curTextDescent = curTextFont.descent;      curTextLeading = curTextFont.leading;      var curContext = drawing.$ensureContext();      curContext.font = curTextFont.css    };    p.textAscent = function() {      return curTextAscent    };    p.textDescent = function() {      return curTextDescent    };    p.textLeading = function(leading) {      curTextLeading = leading    };    p.textAlign = function(xalign, yalign) {      horizontalTextAlignment = xalign;      verticalTextAlignment = yalign || 0    };    function toP5String(obj) {      if (obj instanceof String) return obj;      if (typeof obj === "number") {        if (obj === (0 | obj)) return obj.toString();        return p.nf(obj, 0, 3)      }      if (obj === null || obj === undef) return "";      return obj.toString()    }    Drawing2D.prototype.textWidth = function(str) {      var lines = toP5String(str).split(/\r?\n/g),        width = 0;      var i, linesCount = lines.length;      curContext.font = curTextFont.css;      for (i = 0; i < linesCount; ++i) width = Math.max(width, curTextFont.measureTextWidth(lines[i]));      return width | 0    };    Drawing3D.prototype.textWidth = function(str) {      var lines = toP5String(str).split(/\r?\n/g),        width = 0;      var i, linesCount = lines.length;      if (textcanvas === undef) textcanvas = document.createElement("canvas");      var textContext = textcanvas.getContext("2d");      textContext.font = curTextFont.css;      for (i = 0; i < linesCount; ++i) width = Math.max(width, textContext.measureText(lines[i]).width);      return width | 0    };    p.glyphLook = function(font, chr) {      try {        switch (chr) {        case "1":          return font.one;        case "2":          return font.two;        case "3":          return font.three;        case "4":          return font.four;        case "5":          return font.five;        case "6":          return font.six;        case "7":          return font.seven;        case "8":          return font.eight;        case "9":          return font.nine;        case "0":          return font.zero;        case " ":          return font.space;        case "$":          return font.dollar;        case "!":          return font.exclam;        case '"':          return font.quotedbl;        case "#":          return font.numbersign;        case "%":          return font.percent;        case "&":          return font.ampersand;        case "'":          return font.quotesingle;        case "(":          return font.parenleft;        case ")":          return font.parenright;        case "*":          return font.asterisk;        case "+":          return font.plus;        case ",":          return font.comma;        case "-":          return font.hyphen;        case ".":          return font.period;        case "/":          return font.slash;        case "_":          return font.underscore;        case ":":          return font.colon;        case ";":          return font.semicolon;        case "<":          return font.less;        case "=":          return font.equal;        case ">":          return font.greater;        case "?":          return font.question;        case "@":          return font.at;        case "[":          return font.bracketleft;        case "\\":          return font.backslash;        case "]":          return font.bracketright;        case "^":          return font.asciicircum;        case "`":          return font.grave;        case "{":          return font.braceleft;        case "|":          return font.bar;        case "}":          return font.braceright;        case "~":          return font.asciitilde;        default:          return font[chr]        }      } catch(e) {        Processing.debug(e)      }    };    Drawing2D.prototype.text$line = function(str, x, y, z, align) {      var textWidth = 0,        xOffset = 0;      if (!curTextFont.glyph) {        if (str && "fillText" in curContext) {          if (isFillDirty) {            curContext.fillStyle = p.color.toString(currentFillColor);            isFillDirty = false          }          if (align === 39 || align === 3) {            textWidth = curTextFont.measureTextWidth(str);            if (align === 39) xOffset = -textWidth;            else xOffset = -textWidth / 2          }          curContext.fillText(str, x + xOffset, y)        }      } else {        var font = p.glyphTable[curFontName];        saveContext();        curContext.translate(x, y + curTextSize);        if (align === 39 || align === 3) {          textWidth = font.width(str);          if (align === 39) xOffset = -textWidth;          else xOffset = -textWidth / 2        }        var upem = font.units_per_em,          newScale = 1 / upem * curTextSize;        curContext.scale(newScale, newScale);        for (var i = 0, len = str.length; i < len; i++) try {          p.glyphLook(font, str[i]).draw()        } catch(e) {          Processing.debug(e)        }        restoreContext()      }    };    Drawing3D.prototype.text$line = function(str, x, y, z, align) {      if (textcanvas === undef) textcanvas = document.createElement("canvas");      var oldContext = curContext;      curContext = textcanvas.getContext("2d");      curContext.font = curTextFont.css;      var textWidth = curTextFont.measureTextWidth(str);      textcanvas.width = textWidth;      textcanvas.height = curTextSize;      curContext = textcanvas.getContext("2d");      curContext.font = curTextFont.css;      curContext.textBaseline = "top";      Drawing2D.prototype.text$line(str, 0, 0, 0, 37);      var aspect = textcanvas.width / textcanvas.height;      curContext = oldContext;      curContext.bindTexture(curContext.TEXTURE_2D, textTex);      curContext.texImage2D(curContext.TEXTURE_2D, 0, curContext.RGBA, curContext.RGBA, curContext.UNSIGNED_BYTE, textcanvas);      curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_MAG_FILTER, curContext.LINEAR);      curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_MIN_FILTER, curContext.LINEAR);      curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_WRAP_T, curContext.CLAMP_TO_EDGE);      curContext.texParameteri(curContext.TEXTURE_2D, curContext.TEXTURE_WRAP_S, curContext.CLAMP_TO_EDGE);      var xOffset = 0;      if (align === 39) xOffset = -textWidth;      else if (align === 3) xOffset = -textWidth / 2;      var model = new PMatrix3D;      var scalefactor = curTextSize * 0.5;      model.translate(x + xOffset - scalefactor / 2, y - scalefactor, z);      model.scale(-aspect * scalefactor, -scalefactor, scalefactor);      model.translate(-1, -1, -1);      model.transpose();      var view = new PMatrix3D;      view.scale(1, -1, 1);      view.apply(modelView.array());      view.transpose();      curContext.useProgram(programObject2D);      vertexAttribPointer("aVertex2d", programObject2D, "aVertex", 3, textBuffer);      vertexAttribPointer("aTextureCoord2d", programObject2D, "aTextureCoord", 2, textureBuffer);      uniformi("uSampler2d", programObject2D, "uSampler", [0]);      uniformi("uIsDrawingText2d", programObject2D, "uIsDrawingText", true);      uniformMatrix("uModel2d", programObject2D, "uModel", false, model.array());      uniformMatrix("uView2d", programObject2D, "uView", false, view.array());      uniformf("uColor2d", programObject2D, "uColor", fillStyle);      curContext.bindBuffer(curContext.ELEMENT_ARRAY_BUFFER, indexBuffer);      curContext.drawElements(curContext.TRIANGLES, 6, curContext.UNSIGNED_SHORT, 0)    };    function text$4(str, x, y, z) {      var lines, linesCount;      if (str.indexOf("\n") < 0) {        lines = [str];        linesCount = 1      } else {        lines = str.split(/\r?\n/g);        linesCount = lines.length      }      var yOffset = 0;      if (verticalTextAlignment === 101) yOffset = curTextAscent + curTextDescent;      else if (verticalTextAlignment === 3) yOffset = curTextAscent / 2 - (linesCount - 1) * curTextLeading / 2;      else if (verticalTextAlignment === 102) yOffset = -(curTextDescent + (linesCount - 1) * curTextLeading);      for (var i = 0; i < linesCount; ++i) {        var line = lines[i];        drawing.text$line(line, x, y + yOffset, z, horizontalTextAlignment);        yOffset += curTextLeading      }    }    function text$6(str, x, y, width, height, z) {      if (str.length === 0 || width === 0 || height === 0) return;      if (curTextSize > height) return;      var spaceMark = -1;      var start = 0;      var lineWidth = 0;      var drawCommands = [];      for (var charPos = 0, len = str.length; charPos < len; charPos++) {        var currentChar = str[charPos];        var spaceChar = currentChar === " ";        var letterWidth = curTextFont.measureTextWidth(currentChar);        if (currentChar !== "\n" && lineWidth + letterWidth <= width) {          if (spaceChar) spaceMark = charPos;          lineWidth += letterWidth        } else {          if (spaceMark + 1 === start) if (charPos > 0) spaceMark = charPos;          else return;          if (currentChar === "\n") {            drawCommands.push({              text: str.substring(start, charPos),              width: lineWidth            });            start = charPos + 1          } else {            drawCommands.push({              text: str.substring(start, spaceMark + 1),              width: lineWidth            });            start = spaceMark + 1          }          lineWidth = 0;          charPos = start - 1        }      }      if (start < len) drawCommands.push({        text: str.substring(start),        width: lineWidth      });      var xOffset = 1,        yOffset = curTextAscent;      if (horizontalTextAlignment === 3) xOffset = width / 2;      else if (horizontalTextAlignment === 39) xOffset = width;      var linesCount = drawCommands.length,        visibleLines = Math.min(linesCount, Math.floor(height / curTextLeading));      if (verticalTextAlignment === 101) yOffset = curTextAscent + curTextDescent;      else if (verticalTextAlignment === 3) yOffset = height / 2 - curTextLeading * (visibleLines / 2 - 1);      else if (verticalTextAlignment === 102) yOffset = curTextDescent + curTextLeading;      var command, drawCommand, leading;      for (command = 0; command < linesCount; command++) {        leading = command * curTextLeading;        if (yOffset + leading > height - curTextDescent) break;        drawCommand = drawCommands[command];        drawing.text$line(drawCommand.text, x + xOffset, y + yOffset + leading, z, horizontalTextAlignment)      }    }    p.text = function() {      if (textMode === 5) return;      if (arguments.length === 3) text$4(toP5String(arguments[0]), arguments[1], arguments[2], 0);      else if (arguments.length === 4) text$4(toP5String(arguments[0]), arguments[1], arguments[2], arguments[3]);      else if (arguments.length === 5) text$6(toP5String(arguments[0]), arguments[1], arguments[2], arguments[3], arguments[4], 0);      else if (arguments.length === 6) text$6(toP5String(arguments[0]), arguments[1], arguments[2], arguments[3], arguments[4], arguments[5])    };    p.textMode = function(mode) {      textMode = mode    };    p.loadGlyphs = function(url) {      var x, y, cx, cy, nx, ny, d, a, lastCom, lenC, horiz_adv_x, getXY = "[0-9\\-]+",        path;      var regex = function(needle, hay) {        var i = 0,          results = [],          latest, regexp = new RegExp(needle, "g");        latest = results[i] = regexp.exec(hay);        while (latest) {          i++;          latest = results[i] = regexp.exec(hay)        }        return results      };      var buildPath = function(d) {        var c = regex("[A-Za-z][0-9\\- ]+|Z", d);        var beforePathDraw = function() {          saveContext();          return drawing.$ensureContext()        };        var afterPathDraw = function() {          executeContextFill();          executeContextStroke();          restoreContext()        };        path = "return {draw:function(){var curContext=beforePathDraw();curContext.beginPath();";        x = 0;        y = 0;        cx = 0;        cy = 0;        nx = 0;        ny = 0;        d = 0;        a = 0;        lastCom = "";        lenC = c.length - 1;        for (var j = 0; j < lenC; j++) {          var com = c[j][0],            xy = regex(getXY, com);          switch (com[0]) {          case "M":            x = parseFloat(xy[0][0]);            y = parseFloat(xy[1][0]);            path += "curContext.moveTo(" + x + "," + -y + ");";            break;          case "L":            x = parseFloat(xy[0][0]);            y = parseFloat(xy[1][0]);            path += "curContext.lineTo(" + x + "," + -y + ");";            break;          case "H":            x = parseFloat(xy[0][0]);            path += "curContext.lineTo(" + x + "," + -y + ");";            break;          case "V":            y = parseFloat(xy[0][0]);            path += "curContext.lineTo(" + x + "," + -y + ");";            break;          case "T":            nx = parseFloat(xy[0][0]);            ny = parseFloat(xy[1][0]);            if (lastCom === "Q" || lastCom === "T") {              d = Math.sqrt(Math.pow(x - cx, 2) + Math.pow(cy - y, 2));              a = Math.PI + Math.atan2(cx - x, cy - y);              cx = x + Math.sin(a) * d;              cy = y + Math.cos(a) * d            } else {              cx = x;              cy = y            }            path += "curContext.quadraticCurveTo(" + cx + "," + -cy + "," + nx + "," + -ny + ");";            x = nx;            y = ny;            break;          case "Q":            cx = parseFloat(xy[0][0]);            cy = parseFloat(xy[1][0]);            nx = parseFloat(xy[2][0]);            ny = parseFloat(xy[3][0]);            path += "curContext.quadraticCurveTo(" + cx + "," + -cy + "," + nx + "," + -ny + ");";            x = nx;            y = ny;            break;          case "Z":            path += "curContext.closePath();";            break          }          lastCom = com[0]        }        path += "afterPathDraw();";        path += "curContext.translate(" + horiz_adv_x + ",0);";        path += "}}";        return (new Function("beforePathDraw", "afterPathDraw", path))(beforePathDraw, afterPathDraw)      };      var parseSVGFont = function(svg) {        var font = svg.getElementsByTagName("font");        p.glyphTable[url].horiz_adv_x = font[0].getAttribute("horiz-adv-x");        var font_face = svg.getElementsByTagName("font-face")[0];        p.glyphTable[url].units_per_em = parseFloat(font_face.getAttribute("units-per-em"));        p.glyphTable[url].ascent = parseFloat(font_face.getAttribute("ascent"));        p.glyphTable[url].descent = parseFloat(font_face.getAttribute("descent"));        var glyph = svg.getElementsByTagName("glyph"),          len = glyph.length;        for (var i = 0; i < len; i++) {          var unicode = glyph[i].getAttribute("unicode");          var name = glyph[i].getAttribute("glyph-name");          horiz_adv_x = glyph[i].getAttribute("horiz-adv-x");          if (horiz_adv_x === null) horiz_adv_x = p.glyphTable[url].horiz_adv_x;          d = glyph[i].getAttribute("d");          if (d !== undef) {            path = buildPath(d);            p.glyphTable[url][name] = {              name: name,              unicode: unicode,              horiz_adv_x: horiz_adv_x,              draw: path.draw            }          }        }      };      var loadXML = function() {        var xmlDoc;        try {          xmlDoc = document.implementation.createDocument("", "", null)        } catch(e_fx_op) {          Processing.debug(e_fx_op.message);          return        }        try {          xmlDoc.async = false;          xmlDoc.load(url);          parseSVGFont(xmlDoc.getElementsByTagName("svg")[0])        } catch(e_sf_ch) {          Processing.debug(e_sf_ch);          try {            var xmlhttp = new window.XMLHttpRequest;            xmlhttp.open("GET", url, false);            xmlhttp.send(null);            parseSVGFont(xmlhttp.responseXML.documentElement)          } catch(e) {            Processing.debug(e_sf_ch)          }        }      };      p.glyphTable[url] = {};      loadXML(url);      return p.glyphTable[url]    };    p.param = function(name) {      var attributeName = "data-processing-" + name;      if (curElement.hasAttribute(attributeName)) return curElement.getAttribute(attributeName);      for (var i = 0, len = curElement.childNodes.length; i < len; ++i) {        var item = curElement.childNodes.item(i);        if (item.nodeType !== 1 || item.tagName.toLowerCase() !== "param") continue;        if (item.getAttribute("name") === name) return item.getAttribute("value")      }      if (curSketch.params.hasOwnProperty(name)) return curSketch.params[name];      return null    };    function wireDimensionalFunctions(mode) {      if (mode === "3D") drawing = new Drawing3D;      else if (mode === "2D") drawing = new Drawing2D;      else drawing = new DrawingPre;      for (var i in DrawingPre.prototype) if (DrawingPre.prototype.hasOwnProperty(i) && i.indexOf("$") < 0) p[i] = drawing[i];      drawing.$init()    }    function createDrawingPreFunction(name) {      return function() {        wireDimensionalFunctions("2D");        return drawing[name].apply(this, arguments)      }    }    DrawingPre.prototype.translate = createDrawingPreFunction("translate");    DrawingPre.prototype.transform = createDrawingPreFunction("transform");    DrawingPre.prototype.scale = createDrawingPreFunction("scale");    DrawingPre.prototype.pushMatrix = createDrawingPreFunction("pushMatrix");    DrawingPre.prototype.popMatrix = createDrawingPreFunction("popMatrix");    DrawingPre.prototype.resetMatrix = createDrawingPreFunction("resetMatrix");    DrawingPre.prototype.applyMatrix = createDrawingPreFunction("applyMatrix");    DrawingPre.prototype.rotate = createDrawingPreFunction("rotate");    DrawingPre.prototype.rotateZ = createDrawingPreFunction("rotateZ");    DrawingPre.prototype.shearX = createDrawingPreFunction("shearX");    DrawingPre.prototype.shearY = createDrawingPreFunction("shearY");    DrawingPre.prototype.redraw = createDrawingPreFunction("redraw");    DrawingPre.prototype.toImageData = createDrawingPreFunction("toImageData");    DrawingPre.prototype.ambientLight = createDrawingPreFunction("ambientLight");    DrawingPre.prototype.directionalLight = createDrawingPreFunction("directionalLight");    DrawingPre.prototype.lightFalloff = createDrawingPreFunction("lightFalloff");    DrawingPre.prototype.lightSpecular = createDrawingPreFunction("lightSpecular");    DrawingPre.prototype.pointLight = createDrawingPreFunction("pointLight");    DrawingPre.prototype.noLights = createDrawingPreFunction("noLights");    DrawingPre.prototype.spotLight = createDrawingPreFunction("spotLight");    DrawingPre.prototype.beginCamera = createDrawingPreFunction("beginCamera");    DrawingPre.prototype.endCamera = createDrawingPreFunction("endCamera");    DrawingPre.prototype.frustum = createDrawingPreFunction("frustum");    DrawingPre.prototype.box = createDrawingPreFunction("box");    DrawingPre.prototype.sphere = createDrawingPreFunction("sphere");    DrawingPre.prototype.ambient = createDrawingPreFunction("ambient");    DrawingPre.prototype.emissive = createDrawingPreFunction("emissive");    DrawingPre.prototype.shininess = createDrawingPreFunction("shininess");    DrawingPre.prototype.specular = createDrawingPreFunction("specular");    DrawingPre.prototype.fill = createDrawingPreFunction("fill");    DrawingPre.prototype.stroke = createDrawingPreFunction("stroke");    DrawingPre.prototype.strokeWeight = createDrawingPreFunction("strokeWeight");    DrawingPre.prototype.smooth = createDrawingPreFunction("smooth");    DrawingPre.prototype.noSmooth = createDrawingPreFunction("noSmooth");    DrawingPre.prototype.point = createDrawingPreFunction("point");    DrawingPre.prototype.vertex = createDrawingPreFunction("vertex");    DrawingPre.prototype.endShape = createDrawingPreFunction("endShape");    DrawingPre.prototype.bezierVertex = createDrawingPreFunction("bezierVertex");    DrawingPre.prototype.curveVertex = createDrawingPreFunction("curveVertex");    DrawingPre.prototype.curve = createDrawingPreFunction("curve");    DrawingPre.prototype.line = createDrawingPreFunction("line");    DrawingPre.prototype.bezier = createDrawingPreFunction("bezier");    DrawingPre.prototype.rect = createDrawingPreFunction("rect");    DrawingPre.prototype.ellipse = createDrawingPreFunction("ellipse");    DrawingPre.prototype.background = createDrawingPreFunction("background");    DrawingPre.prototype.image = createDrawingPreFunction("image");    DrawingPre.prototype.textWidth = createDrawingPreFunction("textWidth");    DrawingPre.prototype.text$line = createDrawingPreFunction("text$line");    DrawingPre.prototype.$ensureContext = createDrawingPreFunction("$ensureContext");    DrawingPre.prototype.$newPMatrix = createDrawingPreFunction("$newPMatrix");    DrawingPre.prototype.size = function(aWidth, aHeight, aMode) {      wireDimensionalFunctions(aMode === 2 ? "3D" : "2D");      p.size(aWidth, aHeight, aMode)    };    DrawingPre.prototype.$init = nop;    Drawing2D.prototype.$init = function() {      p.size(p.width, p.height);      curContext.lineCap = "round";      p.noSmooth();      p.disableContextMenu()    };    Drawing3D.prototype.$init = function() {      p.use3DContext = true;      p.disableContextMenu()    };    DrawingShared.prototype.$ensureContext = function() {      return curContext    };    function calculateOffset(curElement, event) {      var element = curElement,        offsetX = 0,        offsetY = 0;      p.pmouseX = p.mouseX;      p.pmouseY = p.mouseY;      if (element.offsetParent) {        do {          offsetX += element.offsetLeft;          offsetY += element.offsetTop        } while ( !! (element = element.offsetParent))      }      element = curElement;      do {        offsetX -= element.scrollLeft || 0;        offsetY -= element.scrollTop || 0      } while ( !! (element = element.parentNode));      offsetX += stylePaddingLeft;      offsetY += stylePaddingTop;      offsetX += styleBorderLeft;      offsetY += styleBorderTop;      offsetX += window.pageXOffset;      offsetY += window.pageYOffset;      return {        "X": offsetX,        "Y": offsetY      }    }    function updateMousePosition(curElement, event) {      var offset = calculateOffset(curElement, event);      p.mouseX = event.pageX - offset.X;      p.mouseY = event.pageY - offset.Y    }    function addTouchEventOffset(t) {      var offset = calculateOffset(t.changedTouches[0].target, t.changedTouches[0]),        i;      for (i = 0; i < t.touches.length; i++) {        var touch = t.touches[i];        touch.offsetX = touch.pageX - offset.X;        touch.offsetY = touch.pageY - offset.Y      }      for (i = 0; i < t.targetTouches.length; i++) {        var targetTouch = t.targetTouches[i];        targetTouch.offsetX = targetTouch.pageX - offset.X;        targetTouch.offsetY = targetTouch.pageY - offset.Y      }      for (i = 0; i < t.changedTouches.length; i++) {        var changedTouch = t.changedTouches[i];        changedTouch.offsetX = changedTouch.pageX - offset.X;        changedTouch.offsetY = changedTouch.pageY - offset.Y      }      return t    }    attachEventHandler(curElement, "touchstart", function(t) {      curElement.setAttribute("style", "-webkit-user-select: none");      curElement.setAttribute("onclick", "void(0)");      curElement.setAttribute("style", "-webkit-tap-highlight-color:rgba(0,0,0,0)");      for (var i = 0, ehl = eventHandlers.length; i < ehl; i++) {        var type = eventHandlers[i].type;        if (type === "mouseout" || type === "mousemove" || type === "mousedown" || type === "mouseup" || type === "DOMMouseScroll" || type === "mousewheel" || type === "touchstart") detachEventHandler(eventHandlers[i])      }      if (p.touchStart !== undef || p.touchMove !== undef || p.touchEnd !== undef || p.touchCancel !== undef) {        attachEventHandler(curElement, "touchstart", function(t) {          if (p.touchStart !== undef) {            t = addTouchEventOffset(t);            p.touchStart(t)          }        });        attachEventHandler(curElement, "touchmove", function(t) {          if (p.touchMove !== undef) {            t.preventDefault();            t = addTouchEventOffset(t);            p.touchMove(t)          }        });        attachEventHandler(curElement, "touchend", function(t) {          if (p.touchEnd !== undef) {            t = addTouchEventOffset(t);            p.touchEnd(t)          }        });        attachEventHandler(curElement, "touchcancel", function(t) {          if (p.touchCancel !== undef) {            t = addTouchEventOffset(t);            p.touchCancel(t)          }        })      } else {        attachEventHandler(curElement, "touchstart", function(e) {          updateMousePosition(curElement, e.touches[0]);          p.__mousePressed = true;          p.mouseDragging = false;          p.mouseButton = 37;          if (typeof p.mousePressed === "function") p.mousePressed()        });        attachEventHandler(curElement, "touchmove", function(e) {          e.preventDefault();          updateMousePosition(curElement, e.touches[0]);          if (typeof p.mouseMoved === "function" && !p.__mousePressed) p.mouseMoved();          if (typeof p.mouseDragged === "function" && p.__mousePressed) {            p.mouseDragged();            p.mouseDragging = true          }        });        attachEventHandler(curElement, "touchend", function(e) {          p.__mousePressed = false;          if (typeof p.mouseClicked === "function" && !p.mouseDragging) p.mouseClicked();          if (typeof p.mouseReleased === "function") p.mouseReleased()        })      }      curElement.dispatchEvent(t)    });    (function() {      var enabled = true,        contextMenu = function(e) {        e.preventDefault();        e.stopPropagation()      };      p.disableContextMenu = function() {        if (!enabled) return;        attachEventHandler(curElement, "contextmenu", contextMenu);        enabled = false      };      p.enableContextMenu = function() {        if (enabled) return;        detachEventHandler({          elem: curElement,          type: "contextmenu",          fn: contextMenu        });        enabled = true      }    })();    attachEventHandler(curElement, "mousemove", function(e) {      updateMousePosition(curElement, e);      if (typeof p.mouseMoved === "function" && !p.__mousePressed) p.mouseMoved();      if (typeof p.mouseDragged === "function" && p.__mousePressed) {        p.mouseDragged();        p.mouseDragging = true      }    });    attachEventHandler(curElement, "mouseout", function(e) {      if (typeof p.mouseOut === "function") p.mouseOut()    });    attachEventHandler(curElement, "mouseover", function(e) {      updateMousePosition(curElement, e);      if (typeof p.mouseOver === "function") p.mouseOver()    });    curElement.onmousedown = function() {      curElement.focus();      return false    };    attachEventHandler(curElement, "mousedown", function(e) {      p.__mousePressed = true;      p.mouseDragging = false;      switch (e.which) {      case 1:        p.mouseButton = 37;        break;      case 2:        p.mouseButton = 3;        break;      case 3:        p.mouseButton = 39;        break      }      if (typeof p.mousePressed === "function") p.mousePressed()    });    attachEventHandler(curElement, "mouseup", function(e) {      p.__mousePressed = false;      if (typeof p.mouseClicked === "function" && !p.mouseDragging) p.mouseClicked();      if (typeof p.mouseReleased === "function") p.mouseReleased()    });    var mouseWheelHandler = function(e) {      var delta = 0;      if (e.wheelDelta) {        delta = e.wheelDelta / 120;        if (window.opera) delta = -delta      } else if (e.detail) delta = -e.detail / 3;      p.mouseScroll = delta;      if (delta && typeof p.mouseScrolled === "function") p.mouseScrolled()    };    attachEventHandler(document, "DOMMouseScroll", mouseWheelHandler);    attachEventHandler(document, "mousewheel", mouseWheelHandler);    if (!curElement.getAttribute("tabindex")) curElement.setAttribute("tabindex", 0);    function getKeyCode(e) {      var code = e.which || e.keyCode;      switch (code) {      case 13:        return 10;      case 91:      case 93:      case 224:        return 157;      case 57392:        return 17;      case 46:        return 127;      case 45:        return 155      }      return code    }    function getKeyChar(e) {      var c = e.which || e.keyCode;      var anyShiftPressed = e.shiftKey || e.ctrlKey || e.altKey || e.metaKey;      switch (c) {      case 13:        c = anyShiftPressed ? 13 : 10;        break;      case 8:        c = anyShiftPressed ? 127 : 8;        break      }      return new Char(c)    }    function suppressKeyEvent(e) {      if (typeof e.preventDefault === "function") e.preventDefault();      else if (typeof e.stopPropagation === "function") e.stopPropagation();      return false    }    function updateKeyPressed() {      var ch;      for (ch in pressedKeysMap) if (pressedKeysMap.hasOwnProperty(ch)) {        p.__keyPressed = true;        return      }      p.__keyPressed = false    }    function resetKeyPressed() {      p.__keyPressed = false;      pressedKeysMap = [];      lastPressedKeyCode = null    }    function simulateKeyTyped(code, c) {      pressedKeysMap[code] = c;      lastPressedKeyCode = null;      p.key = c;      p.keyCode = code;      p.keyPressed();      p.keyCode = 0;      p.keyTyped();      updateKeyPressed()    }    function handleKeydown(e) {      var code = getKeyCode(e);      if (code === 127) {        simulateKeyTyped(code, new Char(127));        return      }      if (codedKeys.indexOf(code) < 0) {        lastPressedKeyCode = code;        return      }      var c = new Char(65535);      p.key = c;      p.keyCode = code;      pressedKeysMap[code] = c;      p.keyPressed();      lastPressedKeyCode = null;      updateKeyPressed();      return suppressKeyEvent(e)    }    function handleKeypress(e) {      if (lastPressedKeyCode === null) return;      var code = lastPressedKeyCode,        c = getKeyChar(e);      simulateKeyTyped(code, c);      return suppressKeyEvent(e)    }    function handleKeyup(e) {      var code = getKeyCode(e),        c = pressedKeysMap[code];      if (c === undef) return;      p.key = c;      p.keyCode = code;      p.keyReleased();      delete pressedKeysMap[code];      updateKeyPressed()    }    if (!pgraphicsMode) {      if (aCode instanceof Processing.Sketch) curSketch = aCode;      else if (typeof aCode === "function") curSketch = new Processing.Sketch(aCode);      else if (!aCode) curSketch = new Processing.Sketch(function() {});      else curSketch = Processing.compile(aCode);      p.externals.sketch = curSketch;      wireDimensionalFunctions();      curElement.onfocus = function() {        p.focused = true      };      curElement.onblur = function() {        p.focused = false;        if (!curSketch.options.globalKeyEvents) resetKeyPressed()      };      if (curSketch.options.pauseOnBlur) {        attachEventHandler(window, "focus", function() {          if (doLoop) p.loop()        });        attachEventHandler(window, "blur", function() {          if (doLoop && loopStarted) {            p.noLoop();            doLoop = true          }          resetKeyPressed()        })      }      var keyTrigger = curSketch.options.globalKeyEvents ? window : curElement;      attachEventHandler(keyTrigger, "keydown", handleKeydown);      attachEventHandler(keyTrigger, "keypress", handleKeypress);      attachEventHandler(keyTrigger, "keyup", handleKeyup);      for (var i in Processing.lib) if (Processing.lib.hasOwnProperty(i)) if (Processing.lib[i].hasOwnProperty("attach")) Processing.lib[i].attach(p);      else if (Processing.lib[i] instanceof Function) Processing.lib[i].call(this);      var retryInterval = 100;      var executeSketch = function(processing) {        if (! (curSketch.imageCache.pending || PFont.preloading.pending(retryInterval))) {          if (window.opera) {            var link, element, operaCache = curSketch.imageCache.operaCache;            for (link in operaCache) if (operaCache.hasOwnProperty(link)) {              element = operaCache[link];              if (element !== null) document.body.removeChild(element);              delete operaCache[link]            }          }          curSketch.attach(processing, defaultScope);          curSketch.onLoad(processing);          if (processing.setup) {            processing.setup();            processing.resetMatrix();            curSketch.onSetup()          }          resetContext();          if (processing.draw) if (!doLoop) processing.redraw();          else processing.loop()        } else window.setTimeout(function() {          executeSketch(processing)        },        retryInterval)      };      addInstance(this);      executeSketch(p)    } else {      curSketch = new Processing.Sketch;      wireDimensionalFunctions();      p.size = function(w, h, render) {        if (render && render === 2) wireDimensionalFunctions("3D");        else wireDimensionalFunctions("2D");        p.size(w, h, render)      }    }  };  Processing.debug = debug;  Processing.prototype = defaultScope;  function getGlobalMembers() {    var names = ["abs", "acos", "alpha", "ambient", "ambientLight", "append",      "applyMatrix", "arc", "arrayCopy", "asin", "atan", "atan2", "background", "beginCamera", "beginDraw", "beginShape", "bezier", "bezierDetail", "bezierPoint", "bezierTangent", "bezierVertex", "binary", "blend", "blendColor", "blit_resize", "blue", "box", "breakShape", "brightness", "camera", "ceil", "Character", "color", "colorMode", "concat", "constrain", "copy", "cos", "createFont", "createGraphics", "createImage", "cursor", "curve", "curveDetail", "curvePoint", "curveTangent", "curveTightness", "curveVertex", "day", "degrees", "directionalLight",      "disableContextMenu", "dist", "draw", "ellipse", "ellipseMode", "emissive", "enableContextMenu", "endCamera", "endDraw", "endShape", "exit", "exp", "expand", "externals", "fill", "filter", "floor", "focused", "frameCount", "frameRate", "frustum", "get", "glyphLook", "glyphTable", "green", "height", "hex", "hint", "hour", "hue", "image", "imageMode", "intersect", "join", "key", "keyCode", "keyPressed", "keyReleased", "keyTyped", "lerp", "lerpColor", "lightFalloff", "lights", "lightSpecular", "line", "link", "loadBytes", "loadFont", "loadGlyphs",      "loadImage", "loadPixels", "loadShape", "loadXML", "loadStrings", "log", "loop", "mag", "map", "match", "matchAll", "max", "millis", "min", "minute", "mix", "modelX", "modelY", "modelZ", "modes", "month", "mouseButton", "mouseClicked", "mouseDragged", "mouseMoved", "mouseOut", "mouseOver", "mousePressed", "mouseReleased", "mouseScroll", "mouseScrolled", "mouseX", "mouseY", "name", "nf", "nfc", "nfp", "nfs", "noCursor", "noFill", "noise", "noiseDetail", "noiseSeed", "noLights", "noLoop", "norm", "normal", "noSmooth", "noStroke", "noTint", "ortho",      "param", "parseBoolean", "parseByte", "parseChar", "parseFloat", "parseInt", "peg", "perspective", "PImage", "pixels", "PMatrix2D", "PMatrix3D", "PMatrixStack", "pmouseX", "pmouseY", "point", "pointLight", "popMatrix", "popStyle", "pow", "print", "printCamera", "println", "printMatrix", "printProjection", "PShape", "PShapeSVG", "pushMatrix", "pushStyle", "quad", "radians", "random", "Random", "randomSeed", "rect", "rectMode", "red", "redraw", "requestImage", "resetMatrix", "reverse", "rotate", "rotateX", "rotateY", "rotateZ", "round", "saturation",      "save", "saveFrame", "saveStrings", "scale", "screenX", "screenY", "screenZ", "second", "set", "setup", "shape", "shapeMode", "shared", "shearX", "shearY", "shininess", "shorten", "sin", "size", "smooth", "sort", "specular", "sphere", "sphereDetail", "splice", "split", "splitTokens", "spotLight", "sq", "sqrt", "status", "str", "stroke", "strokeCap", "strokeJoin", "strokeWeight", "subset", "tan", "text", "textAlign", "textAscent", "textDescent", "textFont", "textLeading", "textMode", "textSize", "texture", "textureMode", "textWidth", "tint", "toImageData",      "touchCancel", "touchEnd", "touchMove", "touchStart", "translate", "transform", "triangle", "trim", "unbinary", "unhex", "updatePixels", "use3DContext", "vertex", "width", "XMLElement", "XML", "year", "__contains", "__equals", "__equalsIgnoreCase", "__frameRate", "__hashCode", "__int_cast", "__instanceof", "__keyPressed", "__mousePressed", "__printStackTrace", "__replace", "__replaceAll", "__replaceFirst", "__toCharArray", "__split", "__codePointAt", "__startsWith", "__endsWith", "__matches"];    var members = {};    var i, l;    for (i = 0, l = names.length; i < l; ++i) members[names[i]] = null;    for (var lib in Processing.lib) if (Processing.lib.hasOwnProperty(lib)) if (Processing.lib[lib].exports) {      var exportedNames = Processing.lib[lib].exports;      for (i = 0, l = exportedNames.length; i < l; ++i) members[exportedNames[i]] = null    }    return members  }  function parseProcessing(code) {    var globalMembers = getGlobalMembers();    function splitToAtoms(code) {      var atoms = [];      var items = code.split(/([\{\[\(\)\]\}])/);      var result = items[0];      var stack = [];      for (var i = 1; i < items.length; i += 2) {        var item = items[i];        if (item === "[" || item === "{" || item === "(") {          stack.push(result);          result = item        } else if (item === "]" || item === "}" || item === ")") {          var kind = item === "}" ? "A" : item === ")" ? "B" : "C";          var index = atoms.length;          atoms.push(result + item);          result = stack.pop() + '"' + kind + (index + 1) + '"'        }        result += items[i + 1]      }      atoms.unshift(result);      return atoms    }    function injectStrings(code, strings) {      return code.replace(/'(\d+)'/g, function(all, index) {        var val = strings[index];        if (val.charAt(0) === "/") return val;        return /^'((?:[^'\\\n])|(?:\\.[0-9A-Fa-f]*))'$/.test(val) ? "(new $p.Character(" + val + "))" : val      })    }    function trimSpaces(string) {      var m1 = /^\s*/.exec(string),        result;      if (m1[0].length === string.length) result = {        left: m1[0],        middle: "",        right: ""      };      else {        var m2 = /\s*$/.exec(string);        result = {          left: m1[0],          middle: string.substring(m1[0].length, m2.index),          right: m2[0]        }      }      result.untrim = function(t) {        return this.left + t + this.right      };      return result    }    function trim(string) {      return string.replace(/^\s+/, "").replace(/\s+$/, "")    }    function appendToLookupTable(table, array) {      for (var i = 0, l = array.length; i < l; ++i) table[array[i]] = null;      return table    }    function isLookupTableEmpty(table) {      for (var i in table) if (table.hasOwnProperty(i)) return false;      return true    }    function getAtomIndex(templ) {      return templ.substring(2, templ.length - 1)    }    var codeWoExtraCr = code.replace(/\r\n?|\n\r/g, "\n");    var strings = [];    var codeWoStrings = codeWoExtraCr.replace(/("(?:[^"\\\n]|\\.)*")|('(?:[^'\\\n]|\\.)*')|(([\[\(=|&!\^:?]\s*)(\/(?![*\/])(?:[^\/\\\n]|\\.)*\/[gim]*)\b)|(\/\/[^\n]*\n)|(\/\*(?:(?!\*\/)(?:.|\n))*\*\/)/g, function(all, quoted, aposed, regexCtx, prefix, regex, singleComment, comment) {      var index;      if (quoted || aposed) {        index = strings.length;        strings.push(all);        return "'" + index + "'"      }      if (regexCtx) {        index = strings.length;        strings.push(regex);        return prefix + "'" + index + "'"      }      return comment !== "" ? " " : "\n"    });    codeWoStrings = codeWoStrings.replace(/__x([0-9A-F]{4})/g, function(all, hexCode) {      return "__x005F_x" + hexCode    });    codeWoStrings = codeWoStrings.replace(/\$/g, "__x0024");    var genericsWereRemoved;    var codeWoGenerics = codeWoStrings;    var replaceFunc = function(all, before, types, after) {      if ( !! before || !!after) return all;      genericsWereRemoved = true;      return ""    };    do {      genericsWereRemoved = false;      codeWoGenerics = codeWoGenerics.replace(/([<]?)<\s*((?:\?|[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)(?:\[\])*(?:\s+(?:extends|super)\s+[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)?(?:\s*,\s*(?:\?|[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)(?:\[\])*(?:\s+(?:extends|super)\s+[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)?)*)\s*>([=]?)/g, replaceFunc)    } while (genericsWereRemoved);    var atoms = splitToAtoms(codeWoGenerics);    var replaceContext;    var declaredClasses = {},      currentClassId, classIdSeed = 0;    function addAtom(text, type) {      var lastIndex = atoms.length;      atoms.push(text);      return '"' + type + lastIndex + '"'    }    function generateClassId() {      return "class" + ++classIdSeed    }    function appendClass(class_, classId, scopeId) {      class_.classId = classId;      class_.scopeId = scopeId;      declaredClasses[classId] = class_    }    var transformClassBody, transformInterfaceBody, transformStatementsBlock, transformStatements, transformMain, transformExpression;    var classesRegex = /\b((?:(?:public|private|final|protected|static|abstract)\s+)*)(class|interface)\s+([A-Za-z_$][\w$]*\b)(\s+extends\s+[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*,\s*[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*\b)*)?(\s+implements\s+[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*,\s*[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*\b)*)?\s*("A\d+")/g;    var methodsRegex = /\b((?:(?:public|private|final|protected|static|abstract|synchronized)\s+)*)((?!(?:else|new|return|throw|function|public|private|protected)\b)[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*"C\d+")*)\s*([A-Za-z_$][\w$]*\b)\s*("B\d+")(\s*throws\s+[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*,\s*[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)*)?\s*("A\d+"|;)/g;    var fieldTest = /^((?:(?:public|private|final|protected|static)\s+)*)((?!(?:else|new|return|throw)\b)[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*"C\d+")*)\s*([A-Za-z_$][\w$]*\b)\s*(?:"C\d+"\s*)*([=,]|$)/;    var cstrsRegex = /\b((?:(?:public|private|final|protected|static|abstract)\s+)*)((?!(?:new|return|throw)\b)[A-Za-z_$][\w$]*\b)\s*("B\d+")(\s*throws\s+[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*,\s*[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)*)?\s*("A\d+")/g;    var attrAndTypeRegex = /^((?:(?:public|private|final|protected|static)\s+)*)((?!(?:new|return|throw)\b)[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*(?:\s*"C\d+")*)\s*/;    var functionsRegex = /\bfunction(?:\s+([A-Za-z_$][\w$]*))?\s*("B\d+")\s*("A\d+")/g;    function extractClassesAndMethods(code) {      var s = code;      s = s.replace(classesRegex, function(all) {        return addAtom(all, "E")      });      s = s.replace(methodsRegex, function(all) {        return addAtom(all, "D")      });      s = s.replace(functionsRegex, function(all) {        return addAtom(all, "H")      });      return s    }    function extractConstructors(code, className) {      var result = code.replace(cstrsRegex, function(all, attr, name, params, throws_, body) {        if (name !== className) return all;        return addAtom(all, "G")      });      return result    }    function AstParam(name) {      this.name = name    }    AstParam.prototype.toString = function() {      return this.name    };    function AstParams(params, methodArgsParam) {      this.params = params;      this.methodArgsParam = methodArgsParam    }    AstParams.prototype.getNames = function() {      var names = [];      for (var i = 0, l = this.params.length; i < l; ++i) names.push(this.params[i].name);      return names    };    AstParams.prototype.prependMethodArgs = function(body) {      if (!this.methodArgsParam) return body;      return "{\nvar " + this.methodArgsParam.name + " = Array.prototype.slice.call(arguments, " + this.params.length + ");\n" + body.substring(1)    };    AstParams.prototype.toString = function() {      if (this.params.length === 0) return "()";      var result = "(";      for (var i = 0, l = this.params.length; i < l; ++i) result += this.params[i] + ", ";      return result.substring(0, result.length - 2) + ")"    };    function transformParams(params) {      var paramsWoPars = trim(params.substring(1, params.length - 1));      var result = [],        methodArgsParam = null;      if (paramsWoPars !== "") {        var paramList = paramsWoPars.split(",");        for (var i = 0; i < paramList.length; ++i) {          var param = /\b([A-Za-z_$][\w$]*\b)(\s*"[ABC][\d]*")*\s*$/.exec(paramList[i]);          if (i === paramList.length - 1 && paramList[i].indexOf("...") >= 0) {            methodArgsParam = new AstParam(param[1]);            break          }          result.push(new AstParam(param[1]))        }      }      return new AstParams(result, methodArgsParam)    }    function preExpressionTransform(expr) {      var s = expr;      s = s.replace(/\bnew\s+([A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)(?:\s*"C\d+")+\s*("A\d+")/g, function(all, type, init) {        return init      });      s = s.replace(/\bnew\s+([A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)(?:\s*"B\d+")\s*("A\d+")/g, function(all, type, init) {        return addAtom(all, "F")      });      s = s.replace(functionsRegex, function(all) {        return addAtom(all, "H")      });      s = s.replace(/\bnew\s+([A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)\s*("C\d+"(?:\s*"C\d+")*)/g, function(all, type, index) {        var args = index.replace(/"C(\d+)"/g, function(all, j) {          return atoms[j]        }).replace(/\[\s*\]/g, "[null]").replace(/\s*\]\s*\[\s*/g, ", ");        var arrayInitializer = "{" + args.substring(1, args.length - 1) + "}";        var createArrayArgs = "('" + type + "', " + addAtom(arrayInitializer, "A") + ")";        return "$p.createJavaArray" + addAtom(createArrayArgs, "B")      });      s = s.replace(/(\.\s*length)\s*"B\d+"/g, "$1");      s = s.replace(/#([0-9A-Fa-f]{6})\b/g, function(all, digits) {        return "0xFF" + digits      });      s = s.replace(/"B(\d+)"(\s*(?:[\w$']|"B))/g, function(all, index, next) {        var atom = atoms[index];        if (!/^\(\s*[A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*\s*(?:"C\d+"\s*)*\)$/.test(atom)) return all;        if (/^\(\s*int\s*\)$/.test(atom)) return "(int)" + next;        var indexParts = atom.split(/"C(\d+)"/g);        if (indexParts.length > 1) if (!/^\[\s*\]$/.test(atoms[indexParts[1]])) return all;        return "" + next      });      s = s.replace(/\(int\)([^,\]\)\}\?\:\*\+\-\/\^\|\%\&\~<\>\=]+)/g, function(all, arg) {        var trimmed = trimSpaces(arg);        return trimmed.untrim("__int_cast(" + trimmed.middle + ")")      });      s = s.replace(/\bsuper(\s*"B\d+")/g, "$$superCstr$1").replace(/\bsuper(\s*\.)/g, "$$super$1");      s = s.replace(/\b0+((\d*)(?:\.[\d*])?(?:[eE][\-\+]?\d+)?[fF]?)\b/, function(all, numberWo0, intPart) {        if (numberWo0 === intPart) return all;        return intPart === "" ? "0" + numberWo0 : numberWo0      });      s = s.replace(/\b(\.?\d+\.?)[fF]\b/g, "$1");      s = s.replace(/([^\s])%([^=\s])/g, "$1 % $2");      s = s.replace(/\b(frameRate|keyPressed|mousePressed)\b(?!\s*"B)/g, "__$1");      s = s.replace(/\b(boolean|byte|char|float|int)\s*"B/g, function(all, name) {        return "parse" + name.substring(0, 1).toUpperCase() + name.substring(1) + '"B'      });      s = s.replace(/\bpixels\b\s*(("C(\d+)")|\.length)?(\s*=(?!=)([^,\]\)\}]+))?/g, function(all, indexOrLength, index, atomIndex, equalsPart, rightSide) {        if (index) {          var atom = atoms[atomIndex];          if (equalsPart) return "pixels.setPixel" + addAtom("(" + atom.substring(1, atom.length - 1) + "," + rightSide + ")", "B");          return "pixels.getPixel" + addAtom("(" + atom.substring(1, atom.length - 1) + ")", "B")        }        if (indexOrLength) return "pixels.getLength" + addAtom("()", "B");        if (equalsPart) return "pixels.set" + addAtom("(" + rightSide + ")", "B");        return "pixels.toArray" + addAtom("()", "B")      });      var repeatJavaReplacement;      function replacePrototypeMethods(all, subject, method, atomIndex) {        var atom = atoms[atomIndex];        repeatJavaReplacement = true;        var trimmed = trimSpaces(atom.substring(1, atom.length - 1));        return "__" + method + (trimmed.middle === "" ? addAtom("(" + subject.replace(/\.\s*$/, "") + ")", "B") : addAtom("(" + subject.replace(/\.\s*$/, "") + "," + trimmed.middle + ")", "B"))      }      do {        repeatJavaReplacement = false;        s = s.replace(/((?:'\d+'|\b[A-Za-z_$][\w$]*\s*(?:"[BC]\d+")*)\s*\.\s*(?:[A-Za-z_$][\w$]*\s*(?:"[BC]\d+"\s*)*\.\s*)*)(replace|replaceAll|replaceFirst|contains|equals|equalsIgnoreCase|hashCode|toCharArray|printStackTrace|split|startsWith|endsWith|codePointAt|matches)\s*"B(\d+)"/g, replacePrototypeMethods)      } while (repeatJavaReplacement);      function replaceInstanceof(all, subject, type) {        repeatJavaReplacement = true;        return "__instanceof" + addAtom("(" + subject + ", " + type + ")", "B")      }      do {        repeatJavaReplacement = false;        s = s.replace(/((?:'\d+'|\b[A-Za-z_$][\w$]*\s*(?:"[BC]\d+")*)\s*(?:\.\s*[A-Za-z_$][\w$]*\s*(?:"[BC]\d+"\s*)*)*)instanceof\s+([A-Za-z_$][\w$]*\s*(?:\.\s*[A-Za-z_$][\w$]*)*)/g, replaceInstanceof)      } while (repeatJavaReplacement);      s = s.replace(/\bthis(\s*"B\d+")/g, "$$constr$1");      return s    }    function AstInlineClass(baseInterfaceName, body) {      this.baseInterfaceName = baseInterfaceName;      this.body = body;      body.owner = this    }    AstInlineClass.prototype.toString = function() {      return "new (" + this.body + ")"    };    function transformInlineClass(class_) {      var m = (new RegExp(/\bnew\s*([A-Za-z_$][\w$]*\s*(?:\.\s*[A-Za-z_$][\w$]*)*)\s*"B\d+"\s*"A(\d+)"/)).exec(class_);      var oldClassId = currentClassId,        newClassId = generateClassId();      currentClassId = newClassId;      var uniqueClassName = m[1] + "$" + newClassId;      var inlineClass = new AstInlineClass(uniqueClassName, transformClassBody(atoms[m[2]], uniqueClassName, "", "implements " + m[1]));      appendClass(inlineClass, newClassId, oldClassId);      currentClassId = oldClassId;      return inlineClass    }    function AstFunction(name, params, body) {      this.name = name;      this.params = params;      this.body = body    }    AstFunction.prototype.toString = function() {      var oldContext = replaceContext;      var names = appendToLookupTable({        "this": null      },      this.params.getNames());      replaceContext = function(subject) {        return names.hasOwnProperty(subject.name) ? subject.name : oldContext(subject)      };      var result = "function";      if (this.name) result += " " + this.name;      var body = this.params.prependMethodArgs(this.body.toString());      result += this.params + " " + body;      replaceContext = oldContext;      return result    };    function transformFunction(class_) {      var m = (new RegExp(/\b([A-Za-z_$][\w$]*)\s*"B(\d+)"\s*"A(\d+)"/)).exec(class_);      return new AstFunction(m[1] !== "function" ? m[1] : null, transformParams(atoms[m[2]]), transformStatementsBlock(atoms[m[3]]))    }    function AstInlineObject(members) {      this.members = members    }    AstInlineObject.prototype.toString = function() {      var oldContext = replaceContext;      replaceContext = function(subject) {        return subject.name === "this" ? "this" : oldContext(subject)      };      var result = "";      for (var i = 0, l = this.members.length; i < l; ++i) {        if (this.members[i].label) result += this.members[i].label + ": ";        result += this.members[i].value.toString() + ", "      }      replaceContext = oldContext;      return result.substring(0, result.length - 2)    };    function transformInlineObject(obj) {      var members = obj.split(",");      for (var i = 0; i < members.length; ++i) {        var label = members[i].indexOf(":");        if (label < 0) members[i] = {          value: transformExpression(members[i])        };        else members[i] = {          label: trim(members[i].substring(0, label)),          value: transformExpression(trim(members[i].substring(label + 1)))        }      }      return new AstInlineObject(members)    }    function expandExpression(expr) {      if (expr.charAt(0) === "(" || expr.charAt(0) === "[") return expr.charAt(0) + expandExpression(expr.substring(1, expr.length - 1)) + expr.charAt(expr.length - 1);      if (expr.charAt(0) === "{") {        if (/^\{\s*(?:[A-Za-z_$][\w$]*|'\d+')\s*:/.test(expr)) return "{" + addAtom(expr.substring(1, expr.length - 1), "I") + "}";        return "[" + expandExpression(expr.substring(1, expr.length - 1)) + "]"      }      var trimmed = trimSpaces(expr);      var result = preExpressionTransform(trimmed.middle);      result = result.replace(/"[ABC](\d+)"/g, function(all, index) {        return expandExpression(atoms[index])      });      return trimmed.untrim(result)    }    function replaceContextInVars(expr) {      return expr.replace(/(\.\s*)?((?:\b[A-Za-z_]|\$)[\w$]*)(\s*\.\s*([A-Za-z_$][\w$]*)(\s*\()?)?/g, function(all, memberAccessSign, identifier, suffix, subMember, callSign) {        if (memberAccessSign) return all;        var subject = {          name: identifier,          member: subMember,          callSign: !!callSign        };        return replaceContext(subject) + (suffix === undef ? "" : suffix)      })    }    function AstExpression(expr, transforms) {      this.expr = expr;      this.transforms = transforms    }    AstExpression.prototype.toString = function() {      var transforms = this.transforms;      var expr = replaceContextInVars(this.expr);      return expr.replace(/"!(\d+)"/g, function(all, index) {        return transforms[index].toString()      })    };    transformExpression = function(expr) {      var transforms = [];      var s = expandExpression(expr);      s = s.replace(/"H(\d+)"/g, function(all, index) {        transforms.push(transformFunction(atoms[index]));        return '"!' + (transforms.length - 1) + '"'      });      s = s.replace(/"F(\d+)"/g, function(all, index) {        transforms.push(transformInlineClass(atoms[index]));        return '"!' + (transforms.length - 1) + '"'      });      s = s.replace(/"I(\d+)"/g, function(all, index) {        transforms.push(transformInlineObject(atoms[index]));        return '"!' + (transforms.length - 1) + '"'      });      return new AstExpression(s, transforms)    };    function AstVarDefinition(name, value, isDefault) {      this.name = name;      this.value = value;      this.isDefault = isDefault    }    AstVarDefinition.prototype.toString = function() {      return this.name + " = " + this.value    };    function transformVarDefinition(def, defaultTypeValue) {      var eqIndex = def.indexOf("=");      var name, value, isDefault;      if (eqIndex < 0) {        name = def;        value = defaultTypeValue;        isDefault = true      } else {        name = def.substring(0, eqIndex);        value = transformExpression(def.substring(eqIndex + 1));        isDefault = false      }      return new AstVarDefinition(trim(name.replace(/(\s*"C\d+")+/g, "")), value, isDefault)    }    function getDefaultValueForType(type) {      if (type === "int" || type === "float") return "0";      if (type === "boolean") return "false";      if (type === "color") return "0x00000000";      return "null"    }    function AstVar(definitions, varType) {      this.definitions = definitions;      this.varType = varType    }    AstVar.prototype.getNames = function() {      var names = [];      for (var i = 0, l = this.definitions.length; i < l; ++i) names.push(this.definitions[i].name);      return names    };    AstVar.prototype.toString = function() {      return "var " + this.definitions.join(",")    };    function AstStatement(expression) {      this.expression = expression    }    AstStatement.prototype.toString = function() {      return this.expression.toString()    };    function transformStatement(statement) {      if (fieldTest.test(statement)) {        var attrAndType = attrAndTypeRegex.exec(statement);        var definitions = statement.substring(attrAndType[0].length).split(",");        var defaultTypeValue = getDefaultValueForType(attrAndType[2]);        for (var i = 0; i < definitions.length; ++i) definitions[i] = transformVarDefinition(definitions[i], defaultTypeValue);        return new AstVar(definitions, attrAndType[2])      }      return new AstStatement(transformExpression(statement))    }    function AstForExpression(initStatement, condition, step) {      this.initStatement = initStatement;      this.condition = condition;      this.step = step    }    AstForExpression.prototype.toString = function() {      return "(" + this.initStatement + "; " + this.condition + "; " + this.step + ")"    };    function AstForInExpression(initStatement, container) {      this.initStatement = initStatement;      this.container = container    }    AstForInExpression.prototype.toString = function() {      var init = this.initStatement.toString();      if (init.indexOf("=") >= 0) init = init.substring(0, init.indexOf("="));      return "(" + init + " in " + this.container + ")"    };    function AstForEachExpression(initStatement, container) {      this.initStatement = initStatement;      this.container = container    }    AstForEachExpression.iteratorId = 0;    AstForEachExpression.prototype.toString = function() {      var init = this.initStatement.toString();      var iterator = "$it" + AstForEachExpression.iteratorId++;      var variableName = init.replace(/^\s*var\s*/, "").split("=")[0];      var initIteratorAndVariable = "var " + iterator + " = new $p.ObjectIterator(" + this.container + "), " + variableName + " = void(0)";      var nextIterationCondition = iterator + ".hasNext() && ((" + variableName + " = " + iterator + ".next()) || true)";      return "(" + initIteratorAndVariable + "; " + nextIterationCondition + ";)"    };    function transformForExpression(expr) {      var content;      if (/\bin\b/.test(expr)) {        content = expr.substring(1, expr.length - 1).split(/\bin\b/g);        return new AstForInExpression(transformStatement(trim(content[0])), transformExpression(content[1]))      }      if (expr.indexOf(":") >= 0 && expr.indexOf(";") < 0) {        content = expr.substring(1, expr.length - 1).split(":");        return new AstForEachExpression(transformStatement(trim(content[0])), transformExpression(content[1]))      }      content = expr.substring(1, expr.length - 1).split(";");      return new AstForExpression(transformStatement(trim(content[0])), transformExpression(content[1]), transformExpression(content[2]))    }    function sortByWeight(array) {      array.sort(function(a, b) {        return b.weight - a.weight      })    }    function AstInnerInterface(name, body, isStatic) {      this.name = name;      this.body = body;      this.isStatic = isStatic;      body.owner = this    }    AstInnerInterface.prototype.toString = function() {      return "" + this.body    };    function AstInnerClass(name, body, isStatic) {      this.name = name;      this.body = body;      this.isStatic = isStatic;      body.owner = this    }    AstInnerClass.prototype.toString = function() {      return "" + this.body    };    function transformInnerClass(class_) {      var m = classesRegex.exec(class_);      classesRegex.lastIndex = 0;      var isStatic = m[1].indexOf("static") >= 0;      var body = atoms[getAtomIndex(m[6])],        innerClass;      var oldClassId = currentClassId,        newClassId = generateClassId();      currentClassId = newClassId;      if (m[2] === "interface") innerClass = new AstInnerInterface(m[3], transformInterfaceBody(body, m[3], m[4]), isStatic);      else innerClass = new AstInnerClass(m[3], transformClassBody(body, m[3], m[4], m[5]), isStatic);      appendClass(innerClass, newClassId, oldClassId);      currentClassId = oldClassId;      return innerClass    }    function AstClassMethod(name, params, body, isStatic) {      this.name = name;      this.params = params;      this.body = body;      this.isStatic = isStatic    }    AstClassMethod.prototype.toString = function() {      var paramNames = appendToLookupTable({},      this.params.getNames());      var oldContext = replaceContext;      replaceContext = function(subject) {        return paramNames.hasOwnProperty(subject.name) ? subject.name : oldContext(subject)      };      var body = this.params.prependMethodArgs(this.body.toString());      var result = "function " + this.methodId + this.params + " " + body + "\n";      replaceContext = oldContext;      return result    };    function transformClassMethod(method) {      var m = methodsRegex.exec(method);      methodsRegex.lastIndex = 0;      var isStatic = m[1].indexOf("static") >= 0;      var body = m[6] !== ";" ? atoms[getAtomIndex(m[6])] : "{}";      return new AstClassMethod(m[3], transformParams(atoms[getAtomIndex(m[4])]), transformStatementsBlock(body), isStatic)    }    function AstClassField(definitions, fieldType, isStatic) {      this.definitions = definitions;      this.fieldType = fieldType;      this.isStatic = isStatic    }    AstClassField.prototype.getNames = function() {      var names = [];      for (var i = 0, l = this.definitions.length; i < l; ++i) names.push(this.definitions[i].name);      return names    };    AstClassField.prototype.toString = function() {      var thisPrefix = replaceContext({        name: "[this]"      });      if (this.isStatic) {        var className = this.owner.name;        var staticDeclarations = [];        for (var i = 0, l = this.definitions.length; i < l; ++i) {          var definition = this.definitions[i];          var name = definition.name,            staticName = className + "." + name;          var declaration = "if(" + staticName + " === void(0)) {\n" + " " + staticName + " = " + definition.value + "; }\n" + "$p.defineProperty(" + thisPrefix + ", " + "'" + name + "', { get: function(){return " + staticName + ";}, " + "set: function(val){" + staticName + " = val;} });\n";          staticDeclarations.push(declaration)        }        return staticDeclarations.join("")      }      return thisPrefix + "." + this.definitions.join("; " + thisPrefix + ".")    };    function transformClassField(statement) {      var attrAndType = attrAndTypeRegex.exec(statement);      var isStatic = attrAndType[1].indexOf("static") >= 0;      var definitions = statement.substring(attrAndType[0].length).split(/,\s*/g);      var defaultTypeValue = getDefaultValueForType(attrAndType[2]);      for (var i = 0; i < definitions.length; ++i) definitions[i] = transformVarDefinition(definitions[i], defaultTypeValue);      return new AstClassField(definitions, attrAndType[2], isStatic)    }    function AstConstructor(params, body) {      this.params = params;      this.body = body    }    AstConstructor.prototype.toString = function() {      var paramNames = appendToLookupTable({},      this.params.getNames());      var oldContext = replaceContext;      replaceContext = function(subject) {        return paramNames.hasOwnProperty(subject.name) ? subject.name : oldContext(subject)      };      var prefix = "function $constr_" + this.params.params.length + this.params.toString();      var body = this.params.prependMethodArgs(this.body.toString());      if (!/\$(superCstr|constr)\b/.test(body)) body = "{\n$superCstr();\n" + body.substring(1);      replaceContext = oldContext;      return prefix + body + "\n"    };    function transformConstructor(cstr) {      var m = (new RegExp(/"B(\d+)"\s*"A(\d+)"/)).exec(cstr);      var params = transformParams(atoms[m[1]]);      return new AstConstructor(params, transformStatementsBlock(atoms[m[2]]))    }    function AstInterfaceBody(name, interfacesNames, methodsNames, fields, innerClasses, misc) {      var i, l;      this.name = name;      this.interfacesNames = interfacesNames;      this.methodsNames = methodsNames;      this.fields = fields;      this.innerClasses = innerClasses;      this.misc = misc;      for (i = 0, l = fields.length; i < l; ++i) fields[i].owner = this    }    AstInterfaceBody.prototype.getMembers = function(classFields, classMethods, classInners) {      if (this.owner.base) this.owner.base.body.getMembers(classFields, classMethods, classInners);      var i, j, l, m;      for (i = 0, l = this.fields.length; i < l; ++i) {        var fieldNames = this.fields[i].getNames();        for (j = 0, m = fieldNames.length; j < m; ++j) classFields[fieldNames[j]] = this.fields[i]      }      for (i = 0, l = this.methodsNames.length; i < l; ++i) {        var methodName = this.methodsNames[i];        classMethods[methodName] = true      }      for (i = 0, l = this.innerClasses.length; i < l; ++i) {        var innerClass = this.innerClasses[i];        classInners[innerClass.name] = innerClass      }    };    AstInterfaceBody.prototype.toString = function() {      function getScopeLevel(p) {        var i = 0;        while (p) {          ++i;          p = p.scope        }        return i      }      var scopeLevel = getScopeLevel(this.owner);      var className = this.name;      var staticDefinitions = "";      var metadata = "";      var thisClassFields = {},        thisClassMethods = {},        thisClassInners = {};      this.getMembers(thisClassFields, thisClassMethods, thisClassInners);      var i, l, j, m;      if (this.owner.interfaces) {        var resolvedInterfaces = [],          resolvedInterface;        for (i = 0, l = this.interfacesNames.length; i < l; ++i) {          if (!this.owner.interfaces[i]) continue;          resolvedInterface = replaceContext({            name: this.interfacesNames[i]          });          resolvedInterfaces.push(resolvedInterface);          staticDefinitions += "$p.extendInterfaceMembers(" + className + ", " + resolvedInterface + ");\n"        }        metadata += className + ".$interfaces = [" + resolvedInterfaces.join(", ") + "];\n"      }      metadata += className + ".$isInterface = true;\n";      metadata += className + ".$methods = ['" + this.methodsNames.join("', '") + "'];\n";      sortByWeight(this.innerClasses);      for (i = 0, l = this.innerClasses.length; i < l; ++i) {        var innerClass = this.innerClasses[i];        if (innerClass.isStatic) staticDefinitions += className + "." + innerClass.name + " = " + innerClass + ";\n"      }      for (i = 0, l = this.fields.length; i < l; ++i) {        var field = this.fields[i];        if (field.isStatic) staticDefinitions += className + "." + field.definitions.join(";\n" + className + ".") + ";\n"      }      return "(function() {\n" + "function " + className + "() { throw 'Unable to create the interface'; }\n" + staticDefinitions + metadata + "return " + className + ";\n" + "})()"    };    transformInterfaceBody = function(body, name, baseInterfaces) {      var declarations = body.substring(1, body.length - 1);      declarations = extractClassesAndMethods(declarations);      declarations = extractConstructors(declarations, name);      var methodsNames = [],        classes = [];      declarations = declarations.replace(/"([DE])(\d+)"/g, function(all, type, index) {        if (type === "D") methodsNames.push(index);        else if (type === "E") classes.push(index);        return ""      });      var fields = declarations.split(/;(?:\s*;)*/g);      var baseInterfaceNames;      var i, l;      if (baseInterfaces !== undef) baseInterfaceNames = baseInterfaces.replace(/^\s*extends\s+(.+?)\s*$/g, "$1").split(/\s*,\s*/g);      for (i = 0, l = methodsNames.length; i < l; ++i) {        var method = transformClassMethod(atoms[methodsNames[i]]);        methodsNames[i] = method.name      }      for (i = 0, l = fields.length - 1; i < l; ++i) {        var field = trimSpaces(fields[i]);        fields[i] = transformClassField(field.middle)      }      var tail = fields.pop();      for (i = 0, l = classes.length; i < l; ++i) classes[i] = transformInnerClass(atoms[classes[i]]);      return new AstInterfaceBody(name, baseInterfaceNames, methodsNames, fields, classes, {        tail: tail      })    };    function AstClassBody(name, baseClassName, interfacesNames, functions, methods, fields, cstrs, innerClasses, misc) {      var i, l;      this.name = name;      this.baseClassName = baseClassName;      this.interfacesNames = interfacesNames;      this.functions = functions;      this.methods = methods;      this.fields = fields;      this.cstrs = cstrs;      this.innerClasses = innerClasses;      this.misc = misc;      for (i = 0, l = fields.length; i < l; ++i) fields[i].owner = this    }    AstClassBody.prototype.getMembers = function(classFields, classMethods, classInners) {      if (this.owner.base) this.owner.base.body.getMembers(classFields, classMethods, classInners);      var i, j, l, m;      for (i = 0, l = this.fields.length; i < l; ++i) {        var fieldNames = this.fields[i].getNames();        for (j = 0, m = fieldNames.length; j < m; ++j) classFields[fieldNames[j]] = this.fields[i]      }      for (i = 0, l = this.methods.length; i < l; ++i) {        var method = this.methods[i];        classMethods[method.name] = method      }      for (i = 0, l = this.innerClasses.length; i < l; ++i) {        var innerClass = this.innerClasses[i];        classInners[innerClass.name] = innerClass      }    };    AstClassBody.prototype.toString = function() {      function getScopeLevel(p) {        var i = 0;        while (p) {          ++i;          p = p.scope        }        return i      }      var scopeLevel = getScopeLevel(this.owner);      var selfId = "$this_" + scopeLevel;      var className = this.name;      var result = "var " + selfId + " = this;\n";      var staticDefinitions = "";      var metadata = "";      var thisClassFields = {},        thisClassMethods = {},        thisClassInners = {};      this.getMembers(thisClassFields, thisClassMethods, thisClassInners);      var oldContext = replaceContext;      replaceContext = function(subject) {        var name = subject.name;        if (name === "this") return subject.callSign || !subject.member ? selfId + ".$self" : selfId;        if (thisClassFields.hasOwnProperty(name)) return thisClassFields[name].isStatic ? className + "." + name : selfId + "." + name;        if (thisClassInners.hasOwnProperty(name)) return selfId + "." + name;        if (thisClassMethods.hasOwnProperty(name)) return thisClassMethods[name].isStatic ? className + "." + name : selfId + ".$self." + name;        return oldContext(subject)      };      var resolvedBaseClassName;      if (this.baseClassName) {        resolvedBaseClassName = oldContext({          name: this.baseClassName        });        result += "var $super = { $upcast: " + selfId + " };\n";        result += "function $superCstr(){" + resolvedBaseClassName + ".apply($super,arguments);if(!('$self' in $super)) $p.extendClassChain($super)}\n";        metadata += className + ".$base = " + resolvedBaseClassName + ";\n"      } else result += "function $superCstr(){$p.extendClassChain(" + selfId + ")}\n";      if (this.owner.base) staticDefinitions += "$p.extendStaticMembers(" + className + ", " + resolvedBaseClassName + ");\n";      var i, l, j, m;      if (this.owner.interfaces) {        var resolvedInterfaces = [],          resolvedInterface;        for (i = 0, l = this.interfacesNames.length; i < l; ++i) {          if (!this.owner.interfaces[i]) continue;          resolvedInterface = oldContext({            name: this.interfacesNames[i]          });          resolvedInterfaces.push(resolvedInterface);          staticDefinitions += "$p.extendInterfaceMembers(" + className + ", " + resolvedInterface + ");\n"        }        metadata += className + ".$interfaces = [" + resolvedInterfaces.join(", ") + "];\n"      }      if (this.functions.length > 0) result += this.functions.join("\n") + "\n";      sortByWeight(this.innerClasses);      for (i = 0, l = this.innerClasses.length; i < l; ++i) {        var innerClass = this.innerClasses[i];        if (innerClass.isStatic) {          staticDefinitions += className + "." + innerClass.name + " = " + innerClass + ";\n";          result += selfId + "." + innerClass.name + " = " + className + "." + innerClass.name + ";\n"        } else result += selfId + "." + innerClass.name + " = " + innerClass + ";\n"      }      for (i = 0, l = this.fields.length; i < l; ++i) {        var field = this.fields[i];        if (field.isStatic) {          staticDefinitions += className + "." + field.definitions.join(";\n" + className + ".") + ";\n";          for (j = 0, m = field.definitions.length; j < m; ++j) {            var fieldName = field.definitions[j].name,              staticName = className + "." + fieldName;            result += "$p.defineProperty(" + selfId + ", '" + fieldName + "', {" + "get: function(){return " + staticName + "}, " + "set: function(val){" + staticName + " = val}});\n"          }        } else result += selfId + "." + field.definitions.join(";\n" + selfId + ".") + ";\n"      }      var methodOverloads = {};      for (i = 0, l = this.methods.length; i < l; ++i) {        var method = this.methods[i];        var overload = methodOverloads[method.name];        var methodId = method.name + "$" + method.params.params.length;        var hasMethodArgs = !!method.params.methodArgsParam;        if (overload) {          ++overload;          methodId += "_" + overload        } else overload = 1;        method.methodId = methodId;        methodOverloads[method.name] = overload;        if (method.isStatic) {          staticDefinitions += method;          staticDefinitions += "$p.addMethod(" + className + ", '" + method.name + "', " + methodId + ", " + hasMethodArgs + ");\n";          result += "$p.addMethod(" + selfId + ", '" + method.name + "', " + methodId + ", " + hasMethodArgs + ");\n"        } else {          result += method;          result += "$p.addMethod(" + selfId + ", '" + method.name + "', " + methodId + ", " + hasMethodArgs + ");\n"        }      }      result += trim(this.misc.tail);      if (this.cstrs.length > 0) result += this.cstrs.join("\n") + "\n";      result += "function $constr() {\n";      var cstrsIfs = [];      for (i = 0, l = this.cstrs.length; i < l; ++i) {        var paramsLength = this.cstrs[i].params.params.length;        var methodArgsPresent = !!this.cstrs[i].params.methodArgsParam;        cstrsIfs.push("if(arguments.length " + (methodArgsPresent ? ">=" : "===") + " " + paramsLength + ") { " + "$constr_" + paramsLength + ".apply(" + selfId + ", arguments); }")      }      if (cstrsIfs.length > 0) result += cstrsIfs.join(" else ") + " else ";      result += "$superCstr();\n}\n";      result += "$constr.apply(null, arguments);\n";      replaceContext = oldContext;      return "(function() {\n" + "function " + className + "() {\n" + result + "}\n" + staticDefinitions + metadata + "return " + className + ";\n" + "})()"    };    transformClassBody = function(body, name, baseName, interfaces) {      var declarations = body.substring(1, body.length - 1);      declarations = extractClassesAndMethods(declarations);      declarations = extractConstructors(declarations, name);      var methods = [],        classes = [],        cstrs = [],        functions = [];      declarations = declarations.replace(/"([DEGH])(\d+)"/g, function(all, type, index) {        if (type === "D") methods.push(index);        else if (type === "E") classes.push(index);        else if (type === "H") functions.push(index);        else cstrs.push(index);        return ""      });      var fields = declarations.replace(/^(?:\s*;)+/, "").split(/;(?:\s*;)*/g);      var baseClassName, interfacesNames;      var i;      if (baseName !== undef) baseClassName = baseName.replace(/^\s*extends\s+([A-Za-z_$][\w$]*\b(?:\s*\.\s*[A-Za-z_$][\w$]*\b)*)\s*$/g, "$1");      if (interfaces !== undef) interfacesNames = interfaces.replace(/^\s*implements\s+(.+?)\s*$/g, "$1").split(/\s*,\s*/g);      for (i = 0; i < functions.length; ++i) functions[i] = transformFunction(atoms[functions[i]]);      for (i = 0; i < methods.length; ++i) methods[i] = transformClassMethod(atoms[methods[i]]);      for (i = 0; i < fields.length - 1; ++i) {        var field = trimSpaces(fields[i]);        fields[i] = transformClassField(field.middle)      }      var tail = fields.pop();      for (i = 0; i < cstrs.length; ++i) cstrs[i] = transformConstructor(atoms[cstrs[i]]);      for (i = 0; i < classes.length; ++i) classes[i] = transformInnerClass(atoms[classes[i]]);      return new AstClassBody(name, baseClassName, interfacesNames, functions, methods, fields, cstrs, classes, {        tail: tail      })    };    function AstInterface(name, body) {      this.name = name;      this.body = body;      body.owner = this    }    AstInterface.prototype.toString = function() {      return "var " + this.name + " = " + this.body + ";\n" + "$p." + this.name + " = " + this.name + ";\n"    };    function AstClass(name, body) {      this.name = name;      this.body = body;      body.owner = this    }    AstClass.prototype.toString = function() {      return "var " + this.name + " = " + this.body + ";\n" + "$p." + this.name + " = " + this.name + ";\n"    };    function transformGlobalClass(class_) {      var m = classesRegex.exec(class_);      classesRegex.lastIndex = 0;      var body = atoms[getAtomIndex(m[6])];      var oldClassId = currentClassId,        newClassId = generateClassId();      currentClassId = newClassId;      var globalClass;      if (m[2] === "interface") globalClass = new AstInterface(m[3], transformInterfaceBody(body, m[3], m[4]));      else globalClass = new AstClass(m[3], transformClassBody(body, m[3], m[4], m[5]));      appendClass(globalClass, newClassId, oldClassId);      currentClassId = oldClassId;      return globalClass    }    function AstMethod(name, params, body) {      this.name = name;      this.params = params;      this.body = body    }    AstMethod.prototype.toString = function() {      var paramNames = appendToLookupTable({},      this.params.getNames());      var oldContext = replaceContext;      replaceContext = function(subject) {        return paramNames.hasOwnProperty(subject.name) ? subject.name : oldContext(subject)      };      var body = this.params.prependMethodArgs(this.body.toString());      var result = "function " + this.name + this.params + " " + body + "\n" + "$p." + this.name + " = " + this.name + ";";      replaceContext = oldContext;      return result    };    function transformGlobalMethod(method) {      var m = methodsRegex.exec(method);      var result = methodsRegex.lastIndex = 0;      return new AstMethod(m[3], transformParams(atoms[getAtomIndex(m[4])]), transformStatementsBlock(atoms[getAtomIndex(m[6])]))    }    function preStatementsTransform(statements) {      var s = statements;      s = s.replace(/\b(catch\s*"B\d+"\s*"A\d+")(\s*catch\s*"B\d+"\s*"A\d+")+/g, "$1");      return s    }    function AstForStatement(argument, misc) {      this.argument = argument;      this.misc = misc    }    AstForStatement.prototype.toString = function() {      return this.misc.prefix + this.argument.toString()    };    function AstCatchStatement(argument, misc) {      this.argument = argument;      this.misc = misc    }    AstCatchStatement.prototype.toString = function() {      return this.misc.prefix + this.argument.toString()    };    function AstPrefixStatement(name, argument, misc) {      this.name = name;      this.argument = argument;      this.misc = misc    }    AstPrefixStatement.prototype.toString = function() {      var result = this.misc.prefix;      if (this.argument !== undef) result += this.argument.toString();      return result    };    function AstSwitchCase(expr) {      this.expr = expr    }    AstSwitchCase.prototype.toString = function() {      return "case " + this.expr + ":"    };    function AstLabel(label) {      this.label = label    }    AstLabel.prototype.toString = function() {      return this.label    };    transformStatements = function(statements, transformMethod, transformClass) {      var nextStatement = new RegExp(/\b(catch|for|if|switch|while|with)\s*"B(\d+)"|\b(do|else|finally|return|throw|try|break|continue)\b|("[ADEH](\d+)")|\b(case)\s+([^:]+):|\b([A-Za-z_$][\w$]*\s*:)|(;)/g);      var res = [];      statements = preStatementsTransform(statements);      var lastIndex = 0,        m, space;      while ((m = nextStatement.exec(statements)) !== null) {        if (m[1] !== undef) {          var i = statements.lastIndexOf('"B', nextStatement.lastIndex);          var statementsPrefix = statements.substring(lastIndex, i);          if (m[1] === "for") res.push(new AstForStatement(transformForExpression(atoms[m[2]]), {            prefix: statementsPrefix          }));          else if (m[1] === "catch") res.push(new AstCatchStatement(transformParams(atoms[m[2]]), {            prefix: statementsPrefix          }));          else res.push(new AstPrefixStatement(m[1], transformExpression(atoms[m[2]]), {            prefix: statementsPrefix          }))        } else if (m[3] !== undef) res.push(new AstPrefixStatement(m[3], undef, {          prefix: statements.substring(lastIndex, nextStatement.lastIndex)        }));        else if (m[4] !== undef) {          space = statements.substring(lastIndex, nextStatement.lastIndex - m[4].length);          if (trim(space).length !== 0) continue;          res.push(space);          var kind = m[4].charAt(1),            atomIndex = m[5];          if (kind === "D") res.push(transformMethod(atoms[atomIndex]));          else if (kind === "E") res.push(transformClass(atoms[atomIndex]));          else if (kind === "H") res.push(transformFunction(atoms[atomIndex]));          else res.push(transformStatementsBlock(atoms[atomIndex]))        } else if (m[6] !== undef) res.push(new AstSwitchCase(transformExpression(trim(m[7]))));        else if (m[8] !== undef) {          space = statements.substring(lastIndex, nextStatement.lastIndex - m[8].length);          if (trim(space).length !== 0) continue;          res.push(new AstLabel(statements.substring(lastIndex, nextStatement.lastIndex)))        } else {          var statement = trimSpaces(statements.substring(lastIndex, nextStatement.lastIndex - 1));          res.push(statement.left);          res.push(transformStatement(statement.middle));          res.push(statement.right + ";")        }        lastIndex = nextStatement.lastIndex      }      var statementsTail = trimSpaces(statements.substring(lastIndex));      res.push(statementsTail.left);      if (statementsTail.middle !== "") {        res.push(transformStatement(statementsTail.middle));        res.push(";" + statementsTail.right)      }      return res    };    function getLocalNames(statements) {      var localNames = [];      for (var i = 0, l = statements.length; i < l; ++i) {        var statement = statements[i];        if (statement instanceof AstVar) localNames = localNames.concat(statement.getNames());        else if (statement instanceof AstForStatement && statement.argument.initStatement instanceof AstVar) localNames = localNames.concat(statement.argument.initStatement.getNames());        else if (statement instanceof AstInnerInterface || statement instanceof AstInnerClass || statement instanceof AstInterface || statement instanceof AstClass || statement instanceof AstMethod || statement instanceof AstFunction) localNames.push(statement.name)      }      return appendToLookupTable({},      localNames)    }    function AstStatementsBlock(statements) {      this.statements = statements    }    AstStatementsBlock.prototype.toString = function() {      var localNames = getLocalNames(this.statements);      var oldContext = replaceContext;      if (!isLookupTableEmpty(localNames)) replaceContext = function(subject) {        return localNames.hasOwnProperty(subject.name) ? subject.name : oldContext(subject)      };      var result = "{\n" + this.statements.join("") + "\n}";      replaceContext = oldContext;      return result    };    transformStatementsBlock = function(block) {      var content = trimSpaces(block.substring(1, block.length - 1));      return new AstStatementsBlock(transformStatements(content.middle))    };    function AstRoot(statements) {      this.statements = statements    }    AstRoot.prototype.toString = function() {      var classes = [],        otherStatements = [],        statement;      for (var i = 0, len = this.statements.length; i < len; ++i) {        statement = this.statements[i];        if (statement instanceof AstClass || statement instanceof AstInterface) classes.push(statement);        else otherStatements.push(statement)      }      sortByWeight(classes);      var localNames = getLocalNames(this.statements);      replaceContext = function(subject) {        var name = subject.name;        if (localNames.hasOwnProperty(name)) return name;        if (globalMembers.hasOwnProperty(name) || PConstants.hasOwnProperty(name) || defaultScope.hasOwnProperty(name)) return "$p." + name;        return name      };      var result = "// this code was autogenerated from PJS\n" + "(function($p) {\n" + classes.join("") + "\n" + otherStatements.join("") + "\n})";      replaceContext = null;      return result    };    transformMain = function() {      var statements = extractClassesAndMethods(atoms[0]);      statements = statements.replace(/\bimport\s+[^;]+;/g, "");      return new AstRoot(transformStatements(statements, transformGlobalMethod, transformGlobalClass))    };    function generateMetadata(ast) {      var globalScope = {};      var id, class_;      for (id in declaredClasses) if (declaredClasses.hasOwnProperty(id)) {        class_ = declaredClasses[id];        var scopeId = class_.scopeId,          name = class_.name;        if (scopeId) {          var scope = declaredClasses[scopeId];          class_.scope = scope;          if (scope.inScope === undef) scope.inScope = {};          scope.inScope[name] = class_        } else globalScope[name] = class_      }      function findInScopes(class_, name) {        var parts = name.split(".");        var currentScope = class_.scope,          found;        while (currentScope) {          if (currentScope.hasOwnProperty(parts[0])) {            found = currentScope[parts[0]];            break          }          currentScope = currentScope.scope        }        if (found === undef) found = globalScope[parts[0]];        for (var i = 1, l = parts.length; i < l && found; ++i) found = found.inScope[parts[i]];        return found      }      for (id in declaredClasses) if (declaredClasses.hasOwnProperty(id)) {        class_ = declaredClasses[id];        var baseClassName = class_.body.baseClassName;        if (baseClassName) {          var parent = findInScopes(class_, baseClassName);          if (parent) {            class_.base = parent;            if (!parent.derived) parent.derived = [];            parent.derived.push(class_)          }        }        var interfacesNames = class_.body.interfacesNames,          interfaces = [],          i, l;        if (interfacesNames && interfacesNames.length > 0) {          for (i = 0, l = interfacesNames.length; i < l; ++i) {            var interface_ = findInScopes(class_, interfacesNames[i]);            interfaces.push(interface_);            if (!interface_) continue;            if (!interface_.derived) interface_.derived = [];            interface_.derived.push(class_)          }          if (interfaces.length > 0) class_.interfaces = interfaces        }      }    }    function setWeight(ast) {      var queue = [],        tocheck = {};      var id, scopeId, class_;      for (id in declaredClasses) if (declaredClasses.hasOwnProperty(id)) {        class_ = declaredClasses[id];        if (!class_.inScope && !class_.derived) {          queue.push(id);          class_.weight = 0        } else {          var dependsOn = [];          if (class_.inScope) for (scopeId in class_.inScope) if (class_.inScope.hasOwnProperty(scopeId)) dependsOn.push(class_.inScope[scopeId]);          if (class_.derived) dependsOn = dependsOn.concat(class_.derived);          tocheck[id] = dependsOn        }      }      function removeDependentAndCheck(targetId, from) {        var dependsOn = tocheck[targetId];        if (!dependsOn) return false;        var i = dependsOn.indexOf(from);        if (i < 0) return false;        dependsOn.splice(i, 1);        if (dependsOn.length > 0) return false;        delete tocheck[targetId];        return true      }      while (queue.length > 0) {        id = queue.shift();        class_ = declaredClasses[id];        if (class_.scopeId && removeDependentAndCheck(class_.scopeId, class_)) {          queue.push(class_.scopeId);          declaredClasses[class_.scopeId].weight = class_.weight + 1        }        if (class_.base && removeDependentAndCheck(class_.base.classId, class_)) {          queue.push(class_.base.classId);          class_.base.weight = class_.weight + 1        }        if (class_.interfaces) {          var i, l;          for (i = 0, l = class_.interfaces.length; i < l; ++i) {            if (!class_.interfaces[i] || !removeDependentAndCheck(class_.interfaces[i].classId, class_)) continue;            queue.push(class_.interfaces[i].classId);            class_.interfaces[i].weight = class_.weight + 1          }        }      }    }    var transformed = transformMain();    generateMetadata(transformed);    setWeight(transformed);    var redendered = transformed.toString();    redendered = redendered.replace(/\s*\n(?:[\t ]*\n)+/g, "\n\n");    redendered = redendered.replace(/__x([0-9A-F]{4})/g, function(all, hexCode) {      return String.fromCharCode(parseInt(hexCode, 16))    });    return injectStrings(redendered, strings)  }  function preprocessCode(aCode, sketch) {    var dm = (new RegExp(/\/\*\s*@pjs\s+((?:[^\*]|\*+[^\*\/])*)\*\//g)).exec(aCode);    if (dm && dm.length === 2) {      var jsonItems = [],        directives = dm.splice(1, 2)[0].replace(/\{([\s\S]*?)\}/g, function() {        return function(all, item) {          jsonItems.push(item);          return "{" + (jsonItems.length - 1) + "}"        }      }()).replace("\n", "").replace("\r", "").split(";");      var clean = function(s) {        return s.replace(/^\s*["']?/, "").replace(/["']?\s*$/, "")      };      for (var i = 0, dl = directives.length; i < dl; i++) {        var pair = directives[i].split("=");        if (pair && pair.length === 2) {          var key = clean(pair[0]),            value = clean(pair[1]),            list = [];          if (key === "preload") {            list = value.split(",");            for (var j = 0, jl = list.length; j < jl; j++) {              var imageName = clean(list[j]);              sketch.imageCache.add(imageName)            }          } else if (key === "font") {            list = value.split(",");            for (var x = 0, xl = list.length; x < xl; x++) {              var fontName = clean(list[x]),                index = /^\{(\d*?)\}$/.exec(fontName);              PFont.preloading.add(index ? JSON.parse("{" + jsonItems[index[1]] + "}") : fontName)            }          } else if (key === "pauseOnBlur") sketch.options.pauseOnBlur = value === "true";          else if (key === "globalKeyEvents") sketch.options.globalKeyEvents = value === "true";          else if (key.substring(0, 6) === "param-") sketch.params[key.substring(6)] = value;          else sketch.options[key] = value        }      }    }    return aCode  }  Processing.compile = function(pdeCode) {    var sketch = new Processing.Sketch;    var code = preprocessCode(pdeCode, sketch);    var compiledPde = parseProcessing(code);    sketch.sourceCode = compiledPde;    return sketch  };  var tinylogLite = function() {    var tinylogLite = {},      undef = "undefined",      func = "function",      False = !1,      True = !0,      logLimit = 512,      log = "log";    if (typeof tinylog !== undef && typeof tinylog[log] === func) tinylogLite[log] = tinylog[log];    else if (typeof document !== undef && !document.fake)(function() {      var doc = document,        $div = "div",        $style = "style",        $title = "title",        containerStyles = {        zIndex: 1E4,        position: "fixed",        bottom: "0px",        width: "100%",        height: "15%",        fontFamily: "sans-serif",        color: "#ccc",        backgroundColor: "black"      },        outputStyles = {        position: "relative",        fontFamily: "monospace",        overflow: "auto",        height: "100%",        paddingTop: "5px"      },        resizerStyles = {        height: "5px",        marginTop: "-5px",        cursor: "n-resize",        backgroundColor: "darkgrey"      },        closeButtonStyles = {        position: "absolute",        top: "5px",        right: "20px",        color: "#111",        MozBorderRadius: "4px",        webkitBorderRadius: "4px",        borderRadius: "4px",        cursor: "pointer",        fontWeight: "normal",        textAlign: "center",        padding: "3px 5px",        backgroundColor: "#333",        fontSize: "12px"      },        entryStyles = {        minHeight: "16px"      },        entryTextStyles = {        fontSize: "12px",        margin: "0 8px 0 8px",        maxWidth: "100%",        whiteSpace: "pre-wrap",        overflow: "auto"      },        view = doc.defaultView,        docElem = doc.documentElement,        docElemStyle = docElem[$style],        setStyles = function() {        var i = arguments.length,          elemStyle, styles, style;        while (i--) {          styles = arguments[i--];          elemStyle = arguments[i][$style];          for (style in styles) if (styles.hasOwnProperty(style)) elemStyle[style] = styles[style]        }      },        observer = function(obj, event, handler) {        if (obj.addEventListener) obj.addEventListener(event, handler, False);        else if (obj.attachEvent) obj.attachEvent("on" + event, handler);        return [obj, event, handler]      },        unobserve = function(obj, event, handler) {        if (obj.removeEventListener) obj.removeEventListener(event, handler, False);        else if (obj.detachEvent) obj.detachEvent("on" + event, handler)      },        clearChildren = function(node) {        var children = node.childNodes,          child = children.length;        while (child--) node.removeChild(children.item(0))      },        append = function(to, elem) {        return to.appendChild(elem)      },        createElement = function(localName) {        return doc.createElement(localName)      },        createTextNode = function(text) {        return doc.createTextNode(text)      },        createLog = tinylogLite[log] = function(message) {        var uninit, originalPadding = docElemStyle.paddingBottom,          container = createElement($div),          containerStyle = container[$style],          resizer = append(container, createElement($div)),          output = append(container, createElement($div)),          closeButton = append(container, createElement($div)),          resizingLog = False,          previousHeight = False,          previousScrollTop = False,          messages = 0,          updateSafetyMargin = function() {          docElemStyle.paddingBottom = container.clientHeight + "px"        },          setContainerHeight = function(height) {          var viewHeight = view.innerHeight,            resizerHeight = resizer.clientHeight;          if (height < 0) height = 0;          else if (height + resizerHeight > viewHeight) height = viewHeight - resizerHeight;          containerStyle.height = height / viewHeight * 100 + "%";          updateSafetyMargin()        },          observers = [observer(doc, "mousemove", function(evt) {          if (resizingLog) {            setContainerHeight(view.innerHeight - evt.clientY);            output.scrollTop = previousScrollTop          }        }), observer(doc, "mouseup", function() {          if (resizingLog) resizingLog = previousScrollTop = False        }), observer(resizer, "dblclick", function(evt) {          evt.preventDefault();          if (previousHeight) {            setContainerHeight(previousHeight);            previousHeight = False          } else {            previousHeight = container.clientHeight;            containerStyle.height = "0px"          }        }), observer(resizer, "mousedown", function(evt) {          evt.preventDefault();          resizingLog = True;          previousScrollTop = output.scrollTop        }), observer(resizer, "contextmenu", function() {          resizingLog = False        }), observer(closeButton, "click", function() {          uninit()        })];        uninit = function() {          var i = observers.length;          while (i--) unobserve.apply(tinylogLite, observers[i]);          docElem.removeChild(container);          docElemStyle.paddingBottom = originalPadding;          clearChildren(output);          clearChildren(container);          tinylogLite[log] = createLog        };        setStyles(container, containerStyles, output, outputStyles, resizer, resizerStyles, closeButton, closeButtonStyles);        closeButton[$title] = "Close Log";        append(closeButton, createTextNode("\u2716"));        resizer[$title] = "Double-click to toggle log minimization";        docElem.insertBefore(container, docElem.firstChild);        tinylogLite[log] = function(message) {          if (messages === logLimit) output.removeChild(output.firstChild);          else messages++;          var entry = append(output, createElement($div)),            entryText = append(entry, createElement($div));          entry[$title] = (new Date).toLocaleTimeString();          setStyles(entry, entryStyles, entryText, entryTextStyles);          append(entryText, createTextNode(message));          output.scrollTop = output.scrollHeight        };        tinylogLite[log](message);        updateSafetyMargin()      }    })();    else if (typeof print === func) tinylogLite[log] = print;    return tinylogLite  }();  Processing.logger = tinylogLite;  Processing.version = "1.4.1";  Processing.lib = {};  Processing.registerLibrary = function(name, desc) {    Processing.lib[name] = desc;    if (desc.hasOwnProperty("init")) desc.init(defaultScope)  };  Processing.instances = processingInstances;  Processing.getInstanceById = function(name) {    return processingInstances[processingInstanceIds[name]]  };  Processing.Sketch = function(attachFunction) {    this.attachFunction = attachFunction;    this.options = {      pauseOnBlur: false,      globalKeyEvents: false    };    this.onLoad = nop;    this.onSetup = nop;    this.onPause = nop;    this.onLoop = nop;    this.onFrameStart = nop;    this.onFrameEnd = nop;    this.onExit = nop;    this.params = {};    this.imageCache = {      pending: 0,      images: {},      operaCache: {},      add: function(href, img) {        if (this.images[href]) return;        if (!isDOMPresent) this.images[href] = null;        if (!img) {          img = new Image;          img.onload = function(owner) {            return function() {              owner.pending--            }          }(this);          this.pending++;          img.src = href        }        this.images[href] = img;        if (window.opera) {          var div = document.createElement("div");          div.appendChild(img);          div.style.position = "absolute";          div.style.opacity = 0;          div.style.width = "1px";          div.style.height = "1px";          if (!this.operaCache[href]) {            document.body.appendChild(div);            this.operaCache[href] = div          }        }      }    };    this.sourceCode = undefined;    this.attach = function(processing) {      if (typeof this.attachFunction === "function") this.attachFunction(processing);      else if (this.sourceCode) {        var func = (new Function("return (" + this.sourceCode + ");"))();        func(processing);        this.attachFunction = func      } else throw "Unable to attach sketch to the processing instance";    };    this.toString = function() {      var i;      var code = "((function(Sketch) {\n";      code += "var sketch = new Sketch(\n" + this.sourceCode + ");\n";      for (i in this.options) if (this.options.hasOwnProperty(i)) {        var value = this.options[i];        code += "sketch.options." + i + " = " + (typeof value === "string" ? '"' + value + '"' : "" + value) + ";\n"      }      for (i in this.imageCache) if (this.options.hasOwnProperty(i)) code += 'sketch.imageCache.add("' + i + '");\n';      code += "return sketch;\n})(Processing.Sketch))";      return code    }  };  var loadSketchFromSources = function(canvas, sources) {    var code = [],      errors = [],      sourcesCount = sources.length,      loaded = 0;    function ajaxAsync(url, callback) {      var xhr = new XMLHttpRequest;      xhr.onreadystatechange = function() {        if (xhr.readyState === 4) {          var error;          if (xhr.status !== 200 && xhr.status !== 0) error = "Invalid XHR status " + xhr.status;          else if (xhr.responseText === "") if ("withCredentials" in new XMLHttpRequest && (new XMLHttpRequest).withCredentials === false && window.location.protocol === "file:") error = "XMLHttpRequest failure, possibly due to a same-origin policy violation. You can try loading this page in another browser, or load it from http://localhost using a local webserver. See the Processing.js README for a more detailed explanation of this problem and solutions.";          else error = "File is empty.";          callback(xhr.responseText, error)        }      };      xhr.open("GET", url, true);      if (xhr.overrideMimeType) xhr.overrideMimeType("application/json");      xhr.setRequestHeader("If-Modified-Since", "Fri, 01 Jan 1960 00:00:00 GMT");      xhr.send(null)    }    function loadBlock(index, filename) {      function callback(block, error) {        code[index] = block;        ++loaded;        if (error) errors.push(filename + " ==> " + error);        if (loaded === sourcesCount) if (errors.length === 0) try {          return new Processing(canvas, code.join("\n"))        } catch(e) {          throw "Processing.js: Unable to execute pjs sketch: " + e;        } else throw "Processing.js: Unable to load pjs sketch files: " + errors.join("\n");      }      if (filename.charAt(0) === "#") {        var scriptElement = document.getElementById(filename.substring(1));        if (scriptElement) callback(scriptElement.text || scriptElement.textContent);        else callback("", "Unable to load pjs sketch: element with id '" + filename.substring(1) + "' was not found");        return      }      ajaxAsync(filename, callback)    }    for (var i = 0; i < sourcesCount; ++i) loadBlock(i, sources[i])  };  var init = function() {    document.removeEventListener("DOMContentLoaded", init, false);    processingInstances = [];    var canvas = document.getElementsByTagName("canvas"),      filenames;    for (var i = 0, l = canvas.length; i < l; i++) {      var processingSources = canvas[i].getAttribute("data-processing-sources");      if (processingSources === null) {        processingSources = canvas[i].getAttribute("data-src");        if (processingSources === null) processingSources = canvas[i].getAttribute("datasrc")      }      if (processingSources) {        filenames = processingSources.split(/\s+/g);        for (var j = 0; j < filenames.length;) if (filenames[j]) j++;        else filenames.splice(j, 1);        loadSketchFromSources(canvas[i], filenames)      }    }    var s, last, source, instance, nodelist = document.getElementsByTagName("script"),      scripts = [];    for (s = nodelist.length - 1; s >= 0; s--) scripts.push(nodelist[s]);    for (s = 0, last = scripts.length; s < last; s++) {      var script = scripts[s];      if (!script.getAttribute) continue;      var type = script.getAttribute("type");      if (type && (type.toLowerCase() === "text/processing" || type.toLowerCase() === "application/processing")) {        var target = script.getAttribute("data-processing-target");        canvas = undef;        if (target) canvas = document.getElementById(target);        else {          var nextSibling = script.nextSibling;          while (nextSibling && nextSibling.nodeType !== 1) nextSibling = nextSibling.nextSibling;          if (nextSibling && nextSibling.nodeName.toLowerCase() === "canvas") canvas = nextSibling        }        if (canvas) {          if (script.getAttribute("src")) {            filenames = script.getAttribute("src").split(/\s+/);            loadSketchFromSources(canvas, filenames);            continue          }          source = script.textContent || script.text;          instance = new Processing(canvas, source)        }      }    }  };  Processing.reload = function() {    if (processingInstances.length > 0) for (var i = processingInstances.length - 1; i >= 0; i--) if (processingInstances[i]) processingInstances[i].exit();    init()  };  Processing.loadSketchFromSources = loadSketchFromSources;  Processing.disableInit = function() {    if (isDOMPresent) document.removeEventListener("DOMContentLoaded", init, false)  };  if (isDOMPresent) {    window["Processing"] = Processing;    document.addEventListener("DOMContentLoaded", init, false)  } else this.Processing = Processing})(window, window.document, Math);
 |