PlaygroundParticlesC.cs 382 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469447044714472447344744475447644774478447944804481448244834484448544864487448844894490449144924493449444954496449744984499450045014502450345044505450645074508450945104511451245134514451545164517451845194520452145224523452445254526452745284529453045314532453345344535453645374538453945404541454245434544454545464547454845494550455145524553455445554556455745584559456045614562456345644565456645674568456945704571457245734574457545764577457845794580458145824583458445854586458745884589459045914592459345944595459645974598459946004601460246034604460546064607460846094610461146124613461446154616461746184619462046214622462346244625462646274628462946304631463246334634463546364637463846394640464146424643464446454646464746484649465046514652465346544655465646574658465946604661466246634664466546664667466846694670467146724673467446754676467746784679468046814682468346844685468646874688468946904691469246934694469546964697469846994700470147024703470447054706470747084709471047114712471347144715471647174718471947204721472247234724472547264727472847294730473147324733473447354736473747384739474047414742474347444745474647474748474947504751475247534754475547564757475847594760476147624763476447654766476747684769477047714772477347744775477647774778477947804781478247834784478547864787478847894790479147924793479447954796479747984799480048014802480348044805480648074808480948104811481248134814481548164817481848194820482148224823482448254826482748284829483048314832483348344835483648374838483948404841484248434844484548464847484848494850485148524853485448554856485748584859486048614862486348644865486648674868486948704871487248734874487548764877487848794880488148824883488448854886488748884889489048914892489348944895489648974898489949004901490249034904490549064907490849094910491149124913491449154916491749184919492049214922492349244925492649274928492949304931493249334934493549364937493849394940494149424943494449454946494749484949495049514952495349544955495649574958495949604961496249634964496549664967496849694970497149724973497449754976497749784979498049814982498349844985498649874988498949904991499249934994499549964997499849995000500150025003500450055006500750085009501050115012501350145015501650175018501950205021502250235024502550265027502850295030503150325033503450355036503750385039504050415042504350445045504650475048504950505051505250535054505550565057505850595060506150625063506450655066506750685069507050715072507350745075507650775078507950805081508250835084508550865087508850895090509150925093509450955096509750985099510051015102510351045105510651075108510951105111511251135114511551165117511851195120512151225123512451255126512751285129513051315132513351345135513651375138513951405141514251435144514551465147514851495150515151525153515451555156515751585159516051615162516351645165516651675168516951705171517251735174517551765177517851795180518151825183518451855186518751885189519051915192519351945195519651975198519952005201520252035204520552065207520852095210521152125213521452155216521752185219522052215222522352245225522652275228522952305231523252335234523552365237523852395240524152425243524452455246524752485249525052515252525352545255525652575258525952605261526252635264526552665267526852695270527152725273527452755276527752785279528052815282528352845285528652875288528952905291529252935294529552965297529852995300530153025303530453055306530753085309531053115312531353145315531653175318531953205321532253235324532553265327532853295330533153325333533453355336533753385339534053415342534353445345534653475348534953505351535253535354535553565357535853595360536153625363536453655366536753685369537053715372537353745375537653775378537953805381538253835384538553865387538853895390539153925393539453955396539753985399540054015402540354045405540654075408540954105411541254135414541554165417541854195420542154225423542454255426542754285429543054315432543354345435543654375438543954405441544254435444544554465447544854495450545154525453545454555456545754585459546054615462546354645465546654675468546954705471547254735474547554765477547854795480548154825483548454855486548754885489549054915492549354945495549654975498549955005501550255035504550555065507550855095510551155125513551455155516551755185519552055215522552355245525552655275528552955305531553255335534553555365537553855395540554155425543554455455546554755485549555055515552555355545555555655575558555955605561556255635564556555665567556855695570557155725573557455755576557755785579558055815582558355845585558655875588558955905591559255935594559555965597559855995600560156025603560456055606560756085609561056115612561356145615561656175618561956205621562256235624562556265627562856295630563156325633563456355636563756385639564056415642564356445645564656475648564956505651565256535654565556565657565856595660566156625663566456655666566756685669567056715672567356745675567656775678567956805681568256835684568556865687568856895690569156925693569456955696569756985699570057015702570357045705570657075708570957105711571257135714571557165717571857195720572157225723572457255726572757285729573057315732573357345735573657375738573957405741574257435744574557465747574857495750575157525753575457555756575757585759576057615762576357645765576657675768576957705771577257735774577557765777577857795780578157825783578457855786578757885789579057915792579357945795579657975798579958005801580258035804580558065807580858095810581158125813581458155816581758185819582058215822582358245825582658275828582958305831583258335834583558365837583858395840584158425843584458455846584758485849585058515852585358545855585658575858585958605861586258635864586558665867586858695870587158725873587458755876587758785879588058815882588358845885588658875888588958905891589258935894589558965897589858995900590159025903590459055906590759085909591059115912591359145915591659175918591959205921592259235924592559265927592859295930593159325933593459355936593759385939594059415942594359445945594659475948594959505951595259535954595559565957595859595960596159625963596459655966596759685969597059715972597359745975597659775978597959805981598259835984598559865987598859895990599159925993599459955996599759985999600060016002600360046005600660076008600960106011601260136014601560166017601860196020602160226023602460256026602760286029603060316032603360346035603660376038603960406041604260436044604560466047604860496050605160526053605460556056605760586059606060616062606360646065606660676068606960706071607260736074607560766077607860796080608160826083608460856086608760886089609060916092609360946095609660976098609961006101610261036104610561066107610861096110611161126113611461156116611761186119612061216122612361246125612661276128612961306131613261336134613561366137613861396140614161426143614461456146614761486149615061516152615361546155615661576158615961606161616261636164616561666167616861696170617161726173617461756176617761786179618061816182618361846185618661876188618961906191619261936194619561966197619861996200620162026203620462056206620762086209621062116212621362146215621662176218621962206221622262236224622562266227622862296230623162326233623462356236623762386239624062416242624362446245624662476248624962506251625262536254625562566257625862596260626162626263626462656266626762686269627062716272627362746275627662776278627962806281628262836284628562866287628862896290629162926293629462956296629762986299630063016302630363046305630663076308630963106311631263136314631563166317631863196320632163226323632463256326632763286329633063316332633363346335633663376338633963406341634263436344634563466347634863496350635163526353635463556356635763586359636063616362636363646365636663676368636963706371637263736374637563766377637863796380638163826383638463856386638763886389639063916392639363946395639663976398639964006401640264036404640564066407640864096410641164126413641464156416641764186419642064216422642364246425642664276428642964306431643264336434643564366437643864396440644164426443644464456446644764486449645064516452645364546455645664576458645964606461646264636464646564666467646864696470647164726473647464756476647764786479648064816482648364846485648664876488648964906491649264936494649564966497649864996500650165026503650465056506650765086509651065116512651365146515651665176518651965206521652265236524652565266527652865296530653165326533653465356536653765386539654065416542654365446545654665476548654965506551655265536554655565566557655865596560656165626563656465656566656765686569657065716572657365746575657665776578657965806581658265836584658565866587658865896590659165926593659465956596659765986599660066016602660366046605660666076608660966106611661266136614661566166617661866196620662166226623662466256626662766286629663066316632663366346635663666376638663966406641664266436644664566466647664866496650665166526653665466556656665766586659666066616662666366646665666666676668666966706671667266736674667566766677667866796680668166826683668466856686668766886689669066916692669366946695669666976698669967006701670267036704670567066707670867096710671167126713671467156716671767186719672067216722672367246725672667276728672967306731673267336734673567366737673867396740674167426743674467456746674767486749675067516752675367546755675667576758675967606761676267636764676567666767676867696770677167726773677467756776677767786779678067816782678367846785678667876788678967906791679267936794679567966797679867996800680168026803680468056806680768086809681068116812681368146815681668176818681968206821682268236824682568266827682868296830683168326833683468356836683768386839684068416842684368446845684668476848684968506851685268536854685568566857685868596860686168626863686468656866686768686869687068716872687368746875687668776878687968806881688268836884688568866887688868896890689168926893689468956896689768986899690069016902690369046905690669076908690969106911691269136914691569166917691869196920692169226923692469256926692769286929693069316932693369346935693669376938693969406941694269436944694569466947694869496950695169526953695469556956695769586959696069616962696369646965696669676968696969706971697269736974697569766977697869796980698169826983698469856986698769886989699069916992699369946995699669976998699970007001700270037004700570067007700870097010701170127013701470157016701770187019702070217022702370247025702670277028702970307031703270337034703570367037703870397040704170427043704470457046704770487049705070517052705370547055705670577058705970607061706270637064706570667067706870697070707170727073707470757076707770787079708070817082708370847085708670877088708970907091709270937094709570967097709870997100710171027103710471057106710771087109711071117112711371147115711671177118711971207121712271237124712571267127712871297130713171327133713471357136713771387139714071417142714371447145714671477148714971507151715271537154715571567157715871597160716171627163716471657166716771687169717071717172717371747175717671777178717971807181718271837184718571867187718871897190719171927193719471957196719771987199720072017202720372047205720672077208720972107211721272137214721572167217721872197220722172227223722472257226722772287229723072317232723372347235723672377238723972407241724272437244724572467247724872497250725172527253725472557256725772587259726072617262726372647265726672677268726972707271727272737274727572767277727872797280728172827283728472857286728772887289729072917292729372947295729672977298729973007301730273037304730573067307730873097310731173127313731473157316731773187319732073217322732373247325732673277328732973307331733273337334733573367337733873397340734173427343734473457346734773487349735073517352735373547355735673577358735973607361736273637364736573667367736873697370737173727373737473757376737773787379738073817382738373847385738673877388738973907391739273937394739573967397739873997400740174027403740474057406740774087409741074117412741374147415741674177418741974207421742274237424742574267427742874297430743174327433743474357436743774387439744074417442744374447445744674477448744974507451745274537454745574567457745874597460746174627463746474657466746774687469747074717472747374747475747674777478747974807481748274837484748574867487
  1. using UnityEngine;
  2. using System;
  3. using System.Collections;
  4. using System.Collections.Generic;
  5. using PlaygroundSplines;
  6. namespace ParticlePlayground {
  7. [RequireComponent (typeof(ParticleSystem))]
  8. /// <summary>
  9. /// The PlaygroundParticlesC class is a Particle Playground system driven by the Playground Manager (PlaygroundC). A Particle Playground system contains settings and data for altering a Shuriken component.
  10. /// </summary>
  11. [ExecuteInEditMode()]
  12. public class PlaygroundParticlesC : MonoBehaviour {
  13. /*************************************************************************************************************************************************
  14. PlaygroundParticlesC variables
  15. *************************************************************************************************************************************************/
  16. // Particle Playground settings
  17. /// <summary>
  18. /// The particle source method for distributing particles upon birth.
  19. /// </summary>
  20. [HideInInspector] public SOURCEC source;
  21. /// <summary>
  22. /// Current active state (when using state as source).
  23. /// </summary>
  24. [HideInInspector] public int activeState;
  25. /// <summary>
  26. /// If emission of particles is active on this PlaygroundParticles.
  27. /// </summary>
  28. [HideInInspector] public bool emit = true;
  29. /// <summary>
  30. /// Should a particle re-emit when reaching the end of its lifetime?
  31. /// </summary>
  32. [HideInInspector] public bool loop = true;
  33. /// <summary>
  34. /// Should particles be removed instantly when you set emit to false?
  35. /// </summary>
  36. [HideInInspector] public bool clearParticlesOnEmissionStop = false;
  37. /// <summary>
  38. /// Should the GameObject of this PlaygroundParticlesC disable when not looping?
  39. /// </summary>
  40. [HideInInspector] public bool disableOnDone = false;
  41. /// <summary>
  42. /// The routine that should run when particle simulation has ended.
  43. /// </summary>
  44. [HideInInspector] public ONDONE disableOnDoneRoutine;
  45. /// <summary>
  46. /// The rate to update this PlaygroundParticles.
  47. /// </summary>
  48. [HideInInspector] public int updateRate = 1;
  49. /// <summary>
  50. /// Determins if this particle system should calculate (can be overrided by PlaygroundC.calculate). This will automatically enable/disable when pauseCalculationWhenInvisible is set to true.
  51. /// </summary>
  52. [HideInInspector] public bool calculate = true;
  53. /// <summary>
  54. /// Calculate the delta movement force of this particle system.
  55. /// </summary>
  56. [HideInInspector] public bool calculateDeltaMovement = true;
  57. /// <summary>
  58. /// The strength to multiply delta movement with.
  59. /// </summary>
  60. [HideInInspector] public float deltaMovementStrength = 10f;
  61. /// <summary>
  62. /// The minimum strength to multiply delta movement with. This will apply when using deltaMovementStrengthValueMethod of RandomBetweenTwoValues.
  63. /// </summary>
  64. [HideInInspector] public float minDeltaMovementStrength = 0;
  65. [HideInInspector] public VALUEMETHOD deltaMovementStrengthValueMethod;
  66. /// <summary>
  67. /// The current world object will change its vertices over time. This produces memory garbage with quantity based upon the mesh vertices.
  68. /// </summary>
  69. [HideInInspector] public bool worldObjectUpdateVertices = false;
  70. /// <summary>
  71. /// The current world object will change its normals over time. This produces memory garbage with quantity based upon the mesh normals.
  72. /// </summary>
  73. [HideInInspector] public bool worldObjectUpdateNormals = false;
  74. /// <summary>
  75. /// The method to set the Lifetime Sorting: Nearest Neighbor/Reversed with.
  76. /// </summary>
  77. [HideInInspector] public NEARESTNEIGHBORORIGINMETHOD nearestNeighborOriginMethod;
  78. /// <summary>
  79. /// The initial source position when using lifetime sorting of Nearest Neighbor/Reversed and nearestNeighborOriginMethod of NEARESTNEIGHBORORIGINMETHOD.SourcePoint.
  80. /// This will let you sort the lifetime from a generated particle birth position.
  81. /// </summary>
  82. [HideInInspector] public int nearestNeighborOrigin = 0;
  83. /// <summary>
  84. /// The initial source position when using lifetime sorting of Nearest Neighbor/Reversed and nearestNeighborOriginMethod of NEARESTNEIGHBORORIGINMETHOD.Vector3.
  85. /// This will let you sort the lifetime from a Vector3 in world space.
  86. /// </summary>
  87. [HideInInspector] public Vector3 nearestNeighborOriginVector3;
  88. /// <summary>
  89. /// The initial source position when using lifetime sorting of Nearest Neighbor/Reversed and nearestNeighborOriginMethod of NEARESTNEIGHBORORIGINMETHOD.Transform.
  90. /// This will let you sort the lifetime from a Transform's position in world space.
  91. /// </summary>
  92. [HideInInspector] public Transform nearestNeighborOriginTransform;
  93. /// <summary>
  94. /// The amount of particles within this PlaygroundParticlesC object.
  95. /// </summary>
  96. [HideInInspector] public int particleCount;
  97. /// <summary>
  98. /// The percentage to emit of particleCount in bursts from this PlaygroundParticles.
  99. /// </summary>
  100. [HideInInspector] public float emissionRate = 1f;
  101. /// <summary>
  102. /// The method to calculate Overflow Offset with.
  103. /// </summary>
  104. [HideInInspector] public OVERFLOWMODEC overflowMode = OVERFLOWMODEC.SourceTransform;
  105. /// <summary>
  106. /// The offset direction and magnitude when particle count exceeds source count.
  107. /// </summary>
  108. [HideInInspector] public Vector3 overflowOffset;
  109. /// <summary>
  110. /// Should source position scattering be applied?
  111. /// </summary>
  112. [HideInInspector] public bool applySourceScatter = false;
  113. /// <summary>
  114. /// The minimum spread of source position scattering.
  115. /// </summary>
  116. [HideInInspector] public Vector3 sourceScatterMin;
  117. /// <summary>
  118. /// The maximum spread of source position scattering.
  119. /// </summary>
  120. [HideInInspector] public Vector3 sourceScatterMax;
  121. /// <summary>
  122. /// The scale of source scatter. This can be changed over time to move the scattering seamlessly.
  123. /// </summary>
  124. [HideInInspector] public Vector3 scatterScale = new Vector3(1f,1f,1f);
  125. [HideInInspector] public MINMAXVECTOR3METHOD sourceScatterMethod;
  126. /// <summary>
  127. /// Sort mode for particle lifetime.
  128. /// </summary>
  129. [HideInInspector] public SORTINGC sorting = SORTINGC.Scrambled;
  130. /// <summary>
  131. /// Custom sorting for particle lifetime (when sorting is set to Custom).
  132. /// </summary>
  133. [HideInInspector] public AnimationCurve lifetimeSorting;
  134. /// <summary>
  135. /// Minimum size of particles.
  136. /// </summary>
  137. [HideInInspector] public float sizeMin = 1f;
  138. /// <summary>
  139. /// Maximum size of particles.
  140. /// </summary>
  141. [HideInInspector] public float sizeMax = 1f;
  142. /// <summary>
  143. /// The scale of minimum and maximum particle size.
  144. /// </summary>
  145. [HideInInspector] public float scale = 1f;
  146. /// <summary>
  147. /// Minimum initial particle rotation.
  148. /// </summary>
  149. [HideInInspector] public float initialRotationMin;
  150. /// <summary>
  151. /// Maximum initial particle rotation.
  152. /// </summary>
  153. [HideInInspector] public float initialRotationMax;
  154. /// <summary>
  155. /// Minimum amount to rotate a particle over time.
  156. /// </summary>
  157. [HideInInspector] public float rotationSpeedMin;
  158. /// <summary>
  159. /// Maximum amount to rotate a particle over time.
  160. /// </summary>
  161. [HideInInspector] public float rotationSpeedMax;
  162. /// <summary>
  163. /// Should the particles rotate towards their movement direction. The rotationNormal will determine from which angle the rotation is based on.
  164. /// </summary>
  165. [HideInInspector] public bool rotateTowardsDirection = false;
  166. /// <summary>
  167. /// The rotation direction normal when rotating towards direction (always normalized value).
  168. /// </summary>
  169. [HideInInspector] public Vector3 rotationNormal = -Vector3.forward;
  170. /// <summary>
  171. /// The method to apply lifetime values.
  172. /// </summary>
  173. [HideInInspector] public VALUEMETHOD lifetimeValueMethod;
  174. /// <summary>
  175. /// The maximum life of a particle in seconds.
  176. /// </summary>
  177. [HideInInspector] public float lifetime;
  178. /// <summary>
  179. /// The minimum life of a particle when using lifetimeValueMethod of RandomBetweenTwoValues.
  180. /// </summary>
  181. [HideInInspector] public float lifetimeMin = 0;
  182. /// <summary>
  183. /// The offset of lifetime in this particle system.
  184. /// </summary>
  185. [HideInInspector] public float lifetimeOffset;
  186. /// <summary>
  187. /// The emission during lifetime. This will compress the Lifetime Sorting pattern over the total lifetime.
  188. /// This can be used to have particles live longer than the otherwise obvious repeating pattern.
  189. /// </summary>
  190. [HideInInspector] public float lifetimeEmission = 1f;
  191. /// <summary>
  192. /// The minimum Shuriken lifetime clamps the lifetime value for each particle.
  193. /// When a Shuriken particle reaches 0 it will result in being removed from screen, where a noticable flicker will occur if the particle has the same birth as death position.
  194. /// Upon using Texture Sheet Animation you may want this to be set to 0 if your particles has a short lifetime (below 1) to get all tiles into the animation.
  195. /// </summary>
  196. [HideInInspector] public float minShurikenLifetime = .08f;
  197. /// <summary>
  198. /// Should lifetime size affect each particle?
  199. /// </summary>
  200. [HideInInspector] public bool applyLifetimeSize = true;
  201. /// <summary>
  202. /// The size over lifetime of each particle.
  203. /// </summary>
  204. [HideInInspector] public AnimationCurve lifetimeSize;
  205. /// <summary>
  206. /// Should particle array size affect each particle? This will multiply the size of each particle depending on its position in the particle array.
  207. /// </summary>
  208. [HideInInspector] public bool applyParticleArraySize;
  209. /// <summary>
  210. /// The size over particle array of each particle. This will multiply the size of each particle depending on its position in the particle array.
  211. /// </summary>
  212. [HideInInspector] public AnimationCurve particleArraySize;
  213. /// <summary>
  214. /// Should the particles transition back to their source position during their lifetime? Use transitionBackToSourceAmount to set the strength by a normalized AnimationCurve.
  215. /// </summary>
  216. [HideInInspector] public bool transitionBackToSource = false;
  217. /// <summary>
  218. /// The amount to transition back to the source position by a normalized AnimationCurve.
  219. /// </summary>
  220. [HideInInspector] public AnimationCurve transitionBackToSourceAmount;
  221. /// <summary>
  222. /// Should the particles only position on their source (and not apply any forces)?
  223. /// </summary>
  224. [HideInInspector] public bool onlySourcePositioning = false;
  225. /// <summary>
  226. /// Should the particles only position by lifetime positioning Vector3AnimationCurves?
  227. /// </summary>
  228. [HideInInspector] public bool onlyLifetimePositioning = false;
  229. /// <summary>
  230. /// The lifetime positioning of particles using a Vector3AnimationCurveC. This will annihilate any forces and only move particles on the X, Y and Z Animation Curves.
  231. /// </summary>
  232. [HideInInspector] public Vector3AnimationCurveC lifetimePositioning;
  233. /// <summary>
  234. /// Should scale over time of lifetime positioning apply?
  235. /// </summary>
  236. [HideInInspector] public bool applyLifetimePositioningTimeScale = false;
  237. /// <summary>
  238. /// Should scale of position of lifetime positioning apply?
  239. /// </summary>
  240. [HideInInspector] public bool applyLifetimePositioningPositionScale = false;
  241. /// <summary>
  242. /// The scale of time for lifetime positioning.
  243. /// </summary>
  244. [HideInInspector] public AnimationCurve lifetimePositioningTimeScale;
  245. /// <summary>
  246. /// The scale of positioning for lifetime positioning.
  247. /// </summary>
  248. [HideInInspector] public AnimationCurve lifetimePositioningPositionScale;
  249. /// <summary>
  250. /// The overall scale of lifetime positioning.
  251. /// </summary>
  252. [HideInInspector] public float lifetimePositioningScale = 1f;
  253. /// <summary>
  254. /// Should lifetime positioning use the direction normal of the source?
  255. /// </summary>
  256. [HideInInspector] public bool lifetimePositioningUsesSourceDirection = false;
  257. /// <summary>
  258. /// Should lifetime velocity affect particles?
  259. /// </summary>
  260. [HideInInspector] public bool applyLifetimeVelocity = false;
  261. /// <summary>
  262. /// The velocity over lifetime of each particle.
  263. /// </summary>
  264. [HideInInspector] public Vector3AnimationCurveC lifetimeVelocity;
  265. /// <summary>
  266. /// The lifetime velocity scale.
  267. /// </summary>
  268. [HideInInspector] public float lifetimeVelocityScale = 1f;
  269. /// <summary>
  270. /// Should initial velocity affect particles?
  271. /// </summary>
  272. [HideInInspector] public bool applyInitialVelocity = false;
  273. /// <summary>
  274. /// The minimum starting velocity of each particle.
  275. /// </summary>
  276. [HideInInspector] public Vector3 initialVelocityMin;
  277. /// <summary>
  278. /// The maximum starting velocity of each particle.
  279. /// </summary>
  280. [HideInInspector] public Vector3 initialVelocityMax;
  281. /// <summary>
  282. /// Determines if the value of initial velocity should be rectangular or spherical.
  283. /// </summary>
  284. [HideInInspector] public MINMAXVECTOR3METHOD initialVelocityMethod;
  285. /// <summary>
  286. /// Should initial local velocity affect particles?
  287. /// </summary>
  288. [HideInInspector] public bool applyInitialLocalVelocity = false;
  289. /// <summary>
  290. /// The minimum starting velocity of each particle with normal or transform direction.
  291. /// </summary>
  292. [HideInInspector] public Vector3 initialLocalVelocityMin;
  293. /// <summary>
  294. /// The maximum starting velocity of each particle with normal or transform direction.
  295. /// </summary>
  296. [HideInInspector] public Vector3 initialLocalVelocityMax;
  297. /// <summary>
  298. /// Determines if the value of initial local velocity should be rectangular or spherical.
  299. /// </summary>
  300. [HideInInspector] public MINMAXVECTOR3METHOD initialLocalVelocityMethod;
  301. /// <summary>
  302. /// Should the initial velocity shape be applied on particle re/birth?
  303. /// </summary>
  304. [HideInInspector] public bool applyInitialVelocityShape = false;
  305. /// <summary>
  306. /// The amount of velocity to apply of the spawning particle's initial/local velocity in form of a Vector3AnimationCurve.
  307. /// </summary>
  308. [HideInInspector] public Vector3AnimationCurveC initialVelocityShape;
  309. /// <summary>
  310. /// The scale of initial velocity shape.
  311. /// </summary>
  312. [HideInInspector] public float initialVelocityShapeScale = 1f;
  313. /// <summary>
  314. /// Should bending affect particles velocity?
  315. /// </summary>
  316. [HideInInspector] public bool applyVelocityBending;
  317. /// <summary>
  318. /// The amount to bend velocity of each particle.
  319. /// </summary>
  320. [HideInInspector] public Vector3 velocityBending;
  321. /// <summary>
  322. /// The type of velocity bending.
  323. /// </summary>
  324. [HideInInspector] public VELOCITYBENDINGTYPEC velocityBendingType;
  325. /// <summary>
  326. /// The constant force towards gravitational vector.
  327. /// </summary>
  328. [HideInInspector] public Vector3 gravity;
  329. /// <summary>
  330. /// The maximum positive- and negative velocity of each particle.
  331. /// </summary>
  332. [HideInInspector] public float maxVelocity = 100f;
  333. /// <summary>
  334. /// The force axis constraints of each particle.
  335. /// </summary>
  336. [HideInInspector] public PlaygroundAxisConstraintsC axisConstraints = new PlaygroundAxisConstraintsC();
  337. /// <summary>
  338. /// Particles inertia over time.
  339. /// </summary>
  340. [HideInInspector] public float damping;
  341. /// <summary>
  342. /// The overall scale of velocity.
  343. /// </summary>
  344. [HideInInspector] public float velocityScale = 1f;
  345. /// <summary>
  346. /// The color over lifetime.
  347. /// </summary>
  348. [HideInInspector] public Gradient lifetimeColor;
  349. /// <summary>
  350. /// The colors over lifetime (if Color Source is set to LifetimeColors).
  351. /// </summary>
  352. [HideInInspector] public List<PlaygroundGradientC> lifetimeColors = new List<PlaygroundGradientC>();
  353. /// <summary>
  354. /// The source to read color from (fallback on Lifetime Color if no source color is available).
  355. /// </summary>
  356. [HideInInspector] public COLORSOURCEC colorSource = COLORSOURCEC.Source;
  357. /// <summary>
  358. /// Should the source color use alpha from Lifetime Color instead of the source's original alpha?
  359. /// </summary>
  360. [HideInInspector] public bool sourceUsesLifetimeAlpha;
  361. /// <summary>
  362. /// The method to color particles with. Either evaluated by their lifetime or by their position in the ParticleCache array.
  363. /// </summary>
  364. [HideInInspector] public COLORMETHOD colorMethod;
  365. /// <summary>
  366. /// Determines if particles should get a secondary alpha applied from the arrayColorAlpha gradient when using colorMethod of COLORMETHOD.ParticleArray.
  367. /// </summary>
  368. [HideInInspector] public bool arrayColorUsesAlpha;
  369. /// <summary>
  370. /// The array color alpha.
  371. /// </summary>
  372. [HideInInspector] public Gradient arrayColorAlpha;
  373. /// <summary>
  374. /// Should the movement of the particle system transform when in local simulation space be compensated for?
  375. /// </summary>
  376. [HideInInspector] public bool applyLocalSpaceMovementCompensation = true;
  377. /// <summary>
  378. /// Should particles get a new random size upon rebirth?
  379. /// </summary>
  380. [HideInInspector] public bool applyRandomSizeOnRebirth = true;
  381. /// <summary>
  382. /// Should particles get a new random velocity upon rebirth?
  383. /// </summary>
  384. [HideInInspector] public bool applyRandomInitialVelocityOnRebirth = true;
  385. /// <summary>
  386. /// Should particles get a new random rotation upon rebirth?
  387. /// </summary>
  388. [HideInInspector] public bool applyRandomRotationOnRebirth = true;
  389. /// <summary>
  390. /// Should particles get a new scatter position upon rebirth?
  391. /// </summary>
  392. [HideInInspector] public bool applyRandomScatterOnRebirth = false;
  393. /// <summary>
  394. /// Should particles get their initial calculated color upon rebirth? (Can resolve flickering upon rebirth.)
  395. /// </summary>
  396. [HideInInspector] public bool applyInitialColorOnRebirth = false;
  397. /// <summary>
  398. /// Should particles get a new random lifetime upon rebirth?
  399. /// </summary>
  400. [HideInInspector] public bool applyRandomLifetimeOnRebirth = true;
  401. /// <summary>
  402. /// Should particle birth position be adjusted with the delta time from birth to current? This will make the particles appear linearly.
  403. /// </summary>
  404. [HideInInspector] public bool applyDeltaOnRebirth = true;
  405. /// <summary>
  406. /// Should each birthing particle calculate the local manipulators immediately?
  407. /// </summary>
  408. [HideInInspector] public bool calculateManipulatorOnRebirth = false;
  409. /// <summary>
  410. /// Should the particle system pause calculation upon becoming invisible?
  411. /// </summary>
  412. [HideInInspector] public bool pauseCalculationWhenInvisible = false;
  413. /// <summary>
  414. /// Should the calculation trigger size gizmo be visible in Scene View?
  415. /// </summary>
  416. [HideInInspector] public bool calculationTriggerSizeGizmo = false;
  417. /// <summary>
  418. /// The calculation trigger transform determines the origin of the rendered particle area. This will affect the calculation trigger when pauseCalculationWhenInvisible is set to true.
  419. /// </summary>
  420. [HideInInspector] public Transform calculationTriggerTransform;
  421. /// <summary>
  422. /// The calculation trigger size determines the size of the rendered particle area. This will affect the calculation trigger when pauseCalculationWhenInvisible is set to true.
  423. /// </summary>
  424. [HideInInspector] public Vector3 calculationTriggerSize = new Vector3(1f,1f,1f);
  425. /// <summary>
  426. /// The calculation trigger offset determines the offset from the calculationTriggerTransform position. This will affect the calculation trigger when pauseCalculationWhenInvisible is set to true.
  427. /// </summary>
  428. [HideInInspector] public Vector3 calculationTriggerOffset = new Vector3();
  429. /// <summary>
  430. /// Should the particle system force Play() when GameObject is outside of camera view? (Fix for Shuriken stop rendering.)
  431. /// </summary>
  432. [HideInInspector] public bool forceVisibilityWhenOutOfFrustrum = true;
  433. /// <summary>
  434. /// Should each particle's position be synced with main-threaad? Use this when dealing with moving source objects or if you experience a laggy particle movement.
  435. /// </summary>
  436. [HideInInspector] public bool syncPositionsOnMainThread = false;
  437. /// <summary>
  438. /// Should the particle system force itself to remain in lockPosition?
  439. /// </summary>
  440. [HideInInspector] public bool applyLockPosition = false;
  441. /// <summary>
  442. /// Should the particle system force itself to remain in lockRotation?
  443. /// </summary>
  444. [HideInInspector] public bool applyLockRotation = false;
  445. /// <summary>
  446. /// Should the particle system force itself to remain in lockScale?
  447. /// </summary>
  448. [HideInInspector] public bool applyLockScale = false;
  449. /// <summary>
  450. /// The locked position is considered local.
  451. /// </summary>
  452. [HideInInspector] public bool lockPositionIsLocal = false;
  453. /// <summary>
  454. /// The locked rotation is considered local.
  455. /// </summary>
  456. [HideInInspector] public bool lockRotationIsLocal = false;
  457. /// <summary>
  458. /// The locked position.
  459. /// </summary>
  460. [HideInInspector] public Vector3 lockPosition = Vector3.zero;
  461. /// <summary>
  462. /// The locked rotation.
  463. /// </summary>
  464. [HideInInspector] public Vector3 lockRotation = Vector3.zero;
  465. /// <summary>
  466. /// The locked scale.
  467. /// </summary>
  468. [HideInInspector] public Vector3 lockScale = new Vector3(1f,1f,1f);
  469. /// <summary>
  470. /// Should the movementCompensationLifetimeStrength affect local space movement compensation?
  471. /// </summary>
  472. [HideInInspector] public bool applyMovementCompensationLifetimeStrength = false;
  473. /// <summary>
  474. /// The strength of movement compensation over particles lifetime
  475. /// </summary>
  476. [HideInInspector] public AnimationCurve movementCompensationLifetimeStrength;
  477. /// <summary>
  478. /// Determines if masking should be applied to the particles.
  479. /// </summary>
  480. [HideInInspector] public bool applyParticleMask;
  481. /// <summary>
  482. /// The masked amount of particles. The particleMaskTime will determine if the particles should fade in/out.
  483. /// </summary>
  484. [HideInInspector] public int particleMask = 0;
  485. /// <summary>
  486. /// The time it takes to mask in/out particles when using particleMask.
  487. /// </summary>
  488. [HideInInspector] public float particleMaskTime = 0f;
  489. /// <summary>
  490. /// The method to distribute the sorting mask when particleMask is above 0.
  491. /// </summary>
  492. [HideInInspector] public MASKSORTINGC particleMaskSorting;
  493. /// <summary>
  494. /// The speed of stretching to reach full effect.
  495. /// </summary>
  496. [HideInInspector] public float stretchSpeed = 1f;
  497. /// <summary>
  498. /// Should the start direction of particle stretching be applied?
  499. /// </summary>
  500. [HideInInspector] public bool applyStretchStartDirection = false;
  501. /// <summary>
  502. /// The starting direction of stretching if all initial velocity is zero.
  503. /// </summary>
  504. [HideInInspector] public Vector3 stretchStartDirection = Vector3.zero;
  505. /// <summary>
  506. /// Should lifetime stretching be applied?
  507. /// </summary>
  508. [HideInInspector] public bool applyLifetimeStretching = false;
  509. /// <summary>
  510. /// The lifetime stretching of stretched particles.
  511. /// </summary>
  512. [HideInInspector] public AnimationCurve stretchLifetime;
  513. /// <summary>
  514. /// The multithreading method how this particle system should calculate. Use this to bypass the Playground Manager's threadMethod.
  515. /// </summary>
  516. [HideInInspector] public ThreadMethodLocal threadMethod;
  517. // Source Script variables
  518. /// <summary>
  519. /// When using Emit() the index will point to the next particle in pool to emit.
  520. /// </summary>
  521. [HideInInspector] public int scriptedEmissionIndex;
  522. /// <summary>
  523. /// When using Emit() the passed in position will determine the position for this particle.
  524. /// </summary>
  525. [HideInInspector] public Vector3 scriptedEmissionPosition;
  526. /// <summary>
  527. /// When using Emit() the passed in velocity will determine the speed and direction for this particle.
  528. /// </summary>
  529. [HideInInspector] public Vector3 scriptedEmissionVelocity;
  530. /// <summary>
  531. /// When using Emit() the passed in color will decide the color for this particle if colorSource is set to COLORSOURCEC.Source.
  532. /// </summary>
  533. [HideInInspector] public Color scriptedEmissionColor = Color.white;
  534. /// <summary>
  535. /// When using Emit() the passed in scriptedLifetime will determine the lifetime of the particle.
  536. /// </summary>
  537. [HideInInspector] float scriptedLifetime = 0;
  538. // Collision detection
  539. /// <summary>
  540. /// Determines if particles can collide. Enable this if you want particles to continuously look for colliders of type collisionType (2D/3D). Particle collision will run on main-thread.
  541. /// </summary>
  542. [HideInInspector] public bool collision = false;
  543. /// <summary>
  544. /// Should particles affect rigidbodies? The mass determines how much they will affect the rigidobdy.
  545. /// </summary>
  546. [HideInInspector] public bool affectRigidbodies = true;
  547. [HideInInspector] public bool inverseRigidbodyCollision = false;
  548. /// <summary>
  549. /// The mass of a particle (calculated in collision with rigidbodies).
  550. /// </summary>
  551. [HideInInspector] public float mass = .01f;
  552. /// <summary>
  553. /// The spherical radius of a particle used upon collision.
  554. /// </summary>
  555. [HideInInspector] public float collisionRadius = 1f;
  556. /// <summary>
  557. /// The layers these particles will collide with.
  558. /// </summary>
  559. [HideInInspector] public LayerMask collisionMask;
  560. [HideInInspector] public List<Transform> collisionExclusion;
  561. /// <summary>
  562. /// The amount a particle will loose of its lifetime on collision.
  563. /// </summary>
  564. [HideInInspector] public float lifetimeLoss = 0f;
  565. /// <summary>
  566. /// The amount a particle will bounce on collision.
  567. /// </summary>
  568. [HideInInspector] public float bounciness = .5f;
  569. /// <summary>
  570. /// The minimum amount of random bounciness (seen as negative offset from the collided surface's normal direction).
  571. /// </summary>
  572. [HideInInspector] public Vector3 bounceRandomMin;
  573. /// <summary>
  574. /// The maximum amount of random bounciness (seen as positive offset from the collided surface's normal direction).
  575. /// </summary>
  576. [HideInInspector] public Vector3 bounceRandomMax;
  577. /// <summary>
  578. /// The Playground Colliders of this particle system. A Playground Collider is an infinite collision plane based on a Transform in the scene.
  579. /// </summary>
  580. [HideInInspector] public List<PlaygroundColliderC> colliders;
  581. /// <summary>
  582. /// The type of collision. This determines if 2D- or 3D raycasting should be used.
  583. /// </summary>
  584. [HideInInspector] public COLLISIONTYPEC collisionType;
  585. /// <summary>
  586. /// Minimum collision depth of Raycast2D.
  587. /// </summary>
  588. [HideInInspector] public float minCollisionDepth = 0f;
  589. /// <summary>
  590. /// Maximum collision depth of Raycast2D.
  591. /// </summary>
  592. [HideInInspector] public float maxCollisionDepth = 0f;
  593. /// <summary>
  594. /// Determines if particles should stick to their collided surface.
  595. /// </summary>
  596. [HideInInspector] public bool stickyCollisions = false;
  597. /// <summary>
  598. /// The sticky collisions offset from the collided surface normal.
  599. /// </summary>
  600. [HideInInspector] public float stickyCollisionsSurfaceOffset = 0;
  601. [HideInInspector] public LayerMask stickyCollisionsMask = -1;
  602. /// <summary>
  603. /// Determines if collisions should be calculated using Vector3.Distance (if set to true) or Vector3.SqrMagnitude (if set to false).
  604. /// </summary>
  605. [HideInInspector] public bool collisionPrecision = false;
  606. /// <summary>
  607. /// Determines if collisions should be cached even if stickyCollisions isn't set to true. Enable this if you want to extend your scripts upon the collision information for the particles when not using sticky collisions.
  608. /// Having this disabled will minimize transform component lookups during collisions.
  609. /// </summary>
  610. [HideInInspector] public bool forceCollisionCaching = false;
  611. // States (source)
  612. /// <summary>
  613. /// The list of States for this PlaygroundParticles. A State is Source data from a texture or mesh which determines where particles will be positioned upon birth.
  614. /// </summary>
  615. public List<ParticleStateC> states = new List<ParticleStateC>();
  616. // Splines
  617. /// <summary>
  618. /// The splines used as Source. Particle birth positions will populate along the spline using the list of particles as normalized time on the curves of the spline. To offset the time use splineTimeOffset.
  619. /// </summary>
  620. [HideInInspector] public List<PlaygroundSpline> splines;
  621. [HideInInspector] public float splineTimeOffset;
  622. [HideInInspector] public bool treatAsOneSpline;
  623. int splineIndex = 0;
  624. // Scene objects (source)
  625. /// <summary>
  626. /// A mesh as source calculated within the scene.
  627. /// </summary>
  628. [HideInInspector] public WorldObject worldObject = new WorldObject();
  629. /// <summary>
  630. /// A skinned mesh as source calculated within the scene.
  631. /// </summary>
  632. [HideInInspector] public SkinnedWorldObject skinnedWorldObject = new SkinnedWorldObject();
  633. [HideInInspector] public bool forceSkinnedMeshUpdateOnMainThread = false;
  634. /// <summary>
  635. /// A transform calculated within the scene.
  636. /// </summary>
  637. [HideInInspector] public Transform sourceTransform;
  638. [HideInInspector] public List<PlaygroundTransformC> sourceTransforms;
  639. [HideInInspector] public bool treatAsOneTransform;
  640. int transformIndex = 0;
  641. // Paint
  642. /// <summary>
  643. /// The paint source of this PlaygroundParticles.
  644. /// </summary>
  645. [HideInInspector] public PaintObjectC paint;
  646. // Projection
  647. /// <summary>
  648. /// The projection source of this PlaygroundParticles.
  649. /// </summary>
  650. [HideInInspector] public ParticleProjectionC projection;
  651. // Manipulators
  652. /// <summary>
  653. /// The list of Local Manipulator Objects handled by this PlaygroundParticlesC object.
  654. /// </summary>
  655. public List<ManipulatorObjectC> manipulators;
  656. // Events
  657. /// <summary>
  658. /// List of event objects handled by this PlaygroundParticlesC object.
  659. /// </summary>
  660. [HideInInspector] public List<PlaygroundEventC> events;
  661. // Cache
  662. /// <summary>
  663. /// Data for each particle.
  664. /// </summary>
  665. [NonSerialized] public PlaygroundCache playgroundCache = new PlaygroundCache();
  666. /// <summary>
  667. /// The particle pool.
  668. /// </summary>
  669. [NonSerialized] public ParticleSystem.Particle[] particleCache;
  670. [NonSerialized] public CollisionCache collisionCache;
  671. // Snapshots
  672. /// <summary>
  673. /// Saved data of properties (positions, velocities, colors etc.).
  674. /// </summary>
  675. [HideInInspector] public List<PlaygroundSave> snapshots = new List<PlaygroundSave>();
  676. /// <summary>
  677. /// Should the particle system load stored data from start?
  678. /// </summary>
  679. [HideInInspector] public bool loadFromStart = false;
  680. /// <summary>
  681. /// Which data should be loaded (if loadFromStart is true).
  682. /// </summary>
  683. [HideInInspector] public int loadFrom = 0;
  684. /// <summary>
  685. /// Should a transition occur whenever a Load is issued?
  686. /// </summary>
  687. [HideInInspector] public bool loadTransition = false;
  688. /// <summary>
  689. /// The type of transition to occur whenever a Load is issued.
  690. /// </summary>
  691. [HideInInspector] public TRANSITIONTYPEC loadTransitionType;
  692. /// <summary>
  693. /// The time for load transition in seconds.
  694. /// </summary>
  695. [HideInInspector] public float loadTransitionTime = 1f;
  696. /// <summary>
  697. /// The storage of position data if this is a snapshot.
  698. /// </summary>
  699. [HideInInspector] public PlaygroundCache snapshotData;
  700. /// <summary>
  701. /// The global time the snapshot was made.
  702. /// </summary>
  703. [HideInInspector] public float timeOfSnapshot = 0;
  704. /// <summary>
  705. /// Is this particle system a snapshot?
  706. /// </summary>
  707. public bool isSnapshot = false;
  708. // Components
  709. /// <summary>
  710. /// This ParticleSystem (Shuriken) component.
  711. /// </summary>
  712. [HideInInspector] public ParticleSystem shurikenParticleSystem;
  713. /// <summary>
  714. /// The id of this PlaygroundParticlesC object.
  715. /// </summary>
  716. [HideInInspector] public int particleSystemId;
  717. /// <summary>
  718. /// The GameObject of a PlaygroundParticlesC object.
  719. /// </summary>
  720. [HideInInspector] public GameObject particleSystemGameObject;
  721. /// <summary>
  722. /// The Transform of a PlaygroundParticlesC object.
  723. /// </summary>
  724. [HideInInspector] public Transform particleSystemTransform;
  725. /// <summary>
  726. /// The Renderer of a PlaygroundParticlesC object.
  727. /// </summary>
  728. [HideInInspector] public Renderer particleSystemRenderer;
  729. /// <summary>
  730. /// The ParticleSystemRenderer of a PlaygroundParticlesC object.
  731. /// </summary>
  732. [HideInInspector] public ParticleSystemRenderer particleSystemRenderer2;
  733. /// <summary>
  734. /// The PlaygroundParticlesC that is controlling this particle system.
  735. /// </summary>
  736. [HideInInspector] public List<PlaygroundParticlesC> eventControlledBy = new List<PlaygroundParticlesC>();
  737. // Turbulence
  738. /// <summary>
  739. /// The Simplex Turbulence object.
  740. /// </summary>
  741. SimplexNoise turbulenceSimplex;
  742. /// <summary>
  743. /// The type of turbulence.
  744. /// </summary>
  745. [HideInInspector] public TURBULENCETYPE turbulenceType = TURBULENCETYPE.None;
  746. /// <summary>
  747. /// The turbulence strength.
  748. /// </summary>
  749. [HideInInspector] public float turbulenceStrength = 10f;
  750. /// <summary>
  751. /// The turbulence resolution scale. A higher value will generate a more dense grid.
  752. /// </summary>
  753. [HideInInspector] public float turbulenceScale = 1f;
  754. /// <summary>
  755. /// The turbulence time scale.
  756. /// </summary>
  757. [HideInInspector] public float turbulenceTimeScale = 1f;
  758. /// <summary>
  759. /// Should Turbulence Lifetime Strength apply?
  760. /// </summary>
  761. [HideInInspector] public bool turbulenceApplyLifetimeStrength = false;
  762. /// <summary>
  763. /// The Turbulence Lifetime Strength. Use this to control how much turbulence will affect the particle over its lifetime.
  764. /// </summary>
  765. [HideInInspector] public AnimationCurve turbulenceLifetimeStrength;
  766. [HideInInspector] public bool isReadyForThreadedCalculations = false;
  767. [HideInInspector] public bool prewarm = false;
  768. [HideInInspector] public float prewarmTime = 1f;
  769. [HideInInspector] public int prewarmCycles = 16;
  770. /// <summary>
  771. /// The simulation time scale for all particles within this system. Set time scale to 0 to pause a particle system.
  772. /// </summary>
  773. [HideInInspector] public float particleTimescale = 1f;
  774. // Internally used variables
  775. int thisLayer;
  776. bool isPrewarming;
  777. bool inTransition = false;
  778. int previousParticleCount = -1;
  779. float previousEmissionRate = 1f;
  780. bool cameFromNonCalculatedFrame = false;
  781. bool cameFromNonEmissionFrame = true;
  782. bool renderModeStretch = false;
  783. float previousSizeMin;
  784. float previousSizeMax;
  785. float previousInitialRotationMin;
  786. float previousInitialRotationMax;
  787. float previousRotationSpeedMin;
  788. float previousRotationSpeedMax;
  789. Vector3 previousVelocityMin;
  790. Vector3 previousVelocityMax;
  791. Vector3 previousLocalVelocityMin;
  792. Vector3 previousLocalVelocityMax;
  793. MINMAXVECTOR3METHOD previousLocalVelocityMethod;
  794. MINMAXVECTOR3METHOD previousVelocityMethod;
  795. SORTINGC previousSorting;
  796. int previousNearestNeighborOrigin;
  797. NEARESTNEIGHBORORIGINMETHOD previousNearestNeighborOriginMethod;
  798. Vector3 previousNearestNeighborOriginVector3;
  799. Transform previousNearestNeighborOriginTransform;
  800. Vector3 nearestNeighborOriginTransformPosition;
  801. MASKSORTINGC previousMaskSorting;
  802. VALUEMETHOD previousLifetimeValueMethod;
  803. float previousLifetime;
  804. float previousLifetimeMin;
  805. float previousLifetimeEmission;
  806. bool previousEmission = true;
  807. bool previousLoop;
  808. float emissionStopped = 0f;
  809. bool queueEmissionHalt = false;
  810. bool hasEmitted = false;
  811. int lifetimeColorId = 0;
  812. System.Random internalRandom01;
  813. float lastTimeUpdated = 0f;
  814. PlaygroundEventParticle eventParticle = new PlaygroundEventParticle();
  815. [HideInInspector] public float localTime = 0f;
  816. [HideInInspector] public float localDeltaTime = 0f;
  817. [HideInInspector] public int previousActiveState;
  818. [HideInInspector] public float simulationStarted;
  819. [HideInInspector] public bool loopExceeded = false;
  820. [HideInInspector] public int loopExceededOnParticle;
  821. bool hasEventManipulatorLocal = false;
  822. bool hasEventManipulatorGlobal = false;
  823. bool hasSeveralManipulatorEvents = false;
  824. bool hasGlobalAffectingManipulators = false;
  825. Quaternion particleSystemRotation;
  826. Quaternion particleSystemInverseRotation;
  827. bool hasActiveParticles = true;
  828. int psTransformNum = -1;
  829. Vector3 manipulatorFix = new Vector3(0,.0001f,0);
  830. PlaygroundParticlesC thisInstance;
  831. float t;
  832. bool cancelDeltaPositioningOnSync;
  833. Quaternion stCompensationRot;
  834. Quaternion stCompensationRotPrev;
  835. Quaternion stRot;
  836. Vector3 stDir;
  837. bool localSpace;
  838. bool overflow;
  839. bool skinnedWorldObjectReady;
  840. bool stateReadyForTextureColor;
  841. int manipulatorEventCount;
  842. bool hasEvent;
  843. bool hasTimerEvent;
  844. bool hasCollisionCache = false;
  845. /*************************************************************************************************************************************************
  846. PlaygroundParticlesC functions
  847. *************************************************************************************************************************************************/
  848. /// <summary>
  849. /// Clones the settings of this Particle Playground system into the passed reference. Note that you additionally need to use CopySaveDataTo() if you want to clone the Snapshots.
  850. /// </summary>
  851. /// <param name="playgroundParticles">Playground particles.</param>
  852. public void CopyTo (PlaygroundParticlesC playgroundParticles) {
  853. // Playground variables
  854. playgroundParticles.source = source;
  855. playgroundParticles.activeState = activeState;
  856. playgroundParticles.emit = emit;
  857. playgroundParticles.loop = loop;
  858. playgroundParticles.clearParticlesOnEmissionStop = clearParticlesOnEmissionStop;
  859. playgroundParticles.disableOnDone = disableOnDone;
  860. playgroundParticles.disableOnDoneRoutine = disableOnDoneRoutine;
  861. playgroundParticles.updateRate = updateRate;
  862. playgroundParticles.calculate = calculate;
  863. playgroundParticles.calculateDeltaMovement = calculateDeltaMovement;
  864. playgroundParticles.deltaMovementStrength = deltaMovementStrength;
  865. playgroundParticles.minDeltaMovementStrength = minDeltaMovementStrength;
  866. playgroundParticles.deltaMovementStrengthValueMethod = deltaMovementStrengthValueMethod;
  867. playgroundParticles.worldObjectUpdateVertices = worldObjectUpdateVertices;
  868. playgroundParticles.worldObjectUpdateNormals = worldObjectUpdateNormals;
  869. playgroundParticles.nearestNeighborOrigin = nearestNeighborOrigin;
  870. playgroundParticles.nearestNeighborOriginMethod = nearestNeighborOriginMethod;
  871. playgroundParticles.nearestNeighborOriginTransform = nearestNeighborOriginTransform;
  872. playgroundParticles.nearestNeighborOriginVector3 = nearestNeighborOriginVector3;
  873. playgroundParticles.particleCount = particleCount;
  874. playgroundParticles.emissionRate = emissionRate;
  875. playgroundParticles.overflowMode = overflowMode;
  876. playgroundParticles.overflowOffset = overflowOffset;
  877. playgroundParticles.applySourceScatter = applySourceScatter;
  878. playgroundParticles.scatterScale = scatterScale;
  879. playgroundParticles.sourceScatterMin = sourceScatterMin;
  880. playgroundParticles.sourceScatterMax = sourceScatterMax;
  881. playgroundParticles.sourceScatterMethod = sourceScatterMethod;
  882. playgroundParticles.sorting = sorting;
  883. playgroundParticles.lifetimeSorting = new AnimationCurve(lifetimeSorting.keys);
  884. playgroundParticles.sizeMin = sizeMin;
  885. playgroundParticles.sizeMax = sizeMax;
  886. playgroundParticles.scale = scale;
  887. playgroundParticles.initialRotationMin = initialRotationMin;
  888. playgroundParticles.initialRotationMax = initialRotationMax;
  889. playgroundParticles.rotationSpeedMin = rotationSpeedMin;
  890. playgroundParticles.rotationSpeedMax = rotationSpeedMax;
  891. playgroundParticles.rotateTowardsDirection = rotateTowardsDirection;
  892. playgroundParticles.rotationNormal = rotationNormal;
  893. playgroundParticles.lifetime = lifetime;
  894. playgroundParticles.lifetimeValueMethod = lifetimeValueMethod;
  895. playgroundParticles.lifetimeMin = lifetimeMin;
  896. playgroundParticles.lifetimeEmission = lifetimeEmission;
  897. playgroundParticles.lifetimeOffset = lifetimeOffset;
  898. playgroundParticles.minShurikenLifetime = minShurikenLifetime;
  899. playgroundParticles.applyLifetimeSize = applyLifetimeSize;
  900. playgroundParticles.lifetimeSize = new AnimationCurve(lifetimeSize.keys);
  901. playgroundParticles.applyParticleArraySize = applyParticleArraySize;
  902. playgroundParticles.particleArraySize = new AnimationCurve(particleArraySize.keys);
  903. playgroundParticles.transitionBackToSource = transitionBackToSource;
  904. playgroundParticles.transitionBackToSourceAmount = new AnimationCurve(transitionBackToSourceAmount.keys);
  905. playgroundParticles.onlySourcePositioning = onlySourcePositioning;
  906. playgroundParticles.onlyLifetimePositioning = onlyLifetimePositioning;
  907. playgroundParticles.lifetimePositioning = lifetimePositioning.Clone();
  908. playgroundParticles.lifetimePositioningScale = lifetimePositioningScale;
  909. playgroundParticles.lifetimePositioningUsesSourceDirection = lifetimePositioningUsesSourceDirection;
  910. playgroundParticles.lifetimePositioningTimeScale = new AnimationCurve(lifetimePositioningTimeScale.keys);
  911. playgroundParticles.lifetimePositioningPositionScale = new AnimationCurve(lifetimePositioningPositionScale.keys);
  912. playgroundParticles.applyLifetimePositioningTimeScale = applyLifetimePositioningTimeScale;
  913. playgroundParticles.applyLifetimePositioningPositionScale = applyLifetimePositioningPositionScale;
  914. playgroundParticles.axisConstraints = axisConstraints.Clone();
  915. playgroundParticles.applyLifetimeVelocity = applyLifetimeVelocity;
  916. playgroundParticles.lifetimeVelocity = lifetimeVelocity.Clone();
  917. playgroundParticles.lifetimeVelocityScale = lifetimeVelocityScale;
  918. playgroundParticles.applyInitialVelocity = applyInitialVelocity;
  919. playgroundParticles.initialVelocityMin = initialVelocityMin;
  920. playgroundParticles.initialVelocityMax = initialVelocityMax;
  921. playgroundParticles.initialVelocityMethod = initialVelocityMethod;
  922. playgroundParticles.applyInitialLocalVelocity = applyInitialLocalVelocity;
  923. playgroundParticles.initialLocalVelocityMin = initialLocalVelocityMin;
  924. playgroundParticles.initialLocalVelocityMax = initialLocalVelocityMax;
  925. playgroundParticles.initialLocalVelocityMethod = initialLocalVelocityMethod;
  926. playgroundParticles.applyVelocityBending = applyVelocityBending;
  927. playgroundParticles.velocityBending = velocityBending;
  928. playgroundParticles.velocityBendingType = velocityBendingType;
  929. playgroundParticles.applyInitialVelocityShape = applyInitialVelocityShape;
  930. playgroundParticles.initialVelocityShape = initialVelocityShape.Clone();
  931. playgroundParticles.initialVelocityShapeScale = initialVelocityShapeScale;
  932. playgroundParticles.gravity = gravity;
  933. playgroundParticles.damping = damping;
  934. playgroundParticles.velocityScale = velocityScale;
  935. playgroundParticles.maxVelocity = maxVelocity;
  936. playgroundParticles.lifetimeColor.SetKeys (lifetimeColor.colorKeys, lifetimeColor.alphaKeys);
  937. playgroundParticles.colorSource = colorSource;
  938. playgroundParticles.sourceUsesLifetimeAlpha = sourceUsesLifetimeAlpha;
  939. playgroundParticles.colorMethod = colorMethod;
  940. playgroundParticles.arrayColorUsesAlpha = arrayColorUsesAlpha;
  941. playgroundParticles.arrayColorAlpha.SetKeys (arrayColorAlpha.colorKeys, arrayColorAlpha.alphaKeys);
  942. playgroundParticles.applyLocalSpaceMovementCompensation = applyLocalSpaceMovementCompensation;
  943. playgroundParticles.applyRandomSizeOnRebirth = applyRandomSizeOnRebirth;
  944. playgroundParticles.applyRandomInitialVelocityOnRebirth = applyRandomInitialVelocityOnRebirth;
  945. playgroundParticles.applyRandomRotationOnRebirth = applyRandomRotationOnRebirth;
  946. playgroundParticles.applyRandomScatterOnRebirth = applyRandomScatterOnRebirth;
  947. playgroundParticles.applyInitialColorOnRebirth = applyInitialColorOnRebirth;
  948. playgroundParticles.applyRandomLifetimeOnRebirth = applyRandomLifetimeOnRebirth;
  949. playgroundParticles.applyDeltaOnRebirth = applyDeltaOnRebirth;
  950. playgroundParticles.calculateManipulatorOnRebirth = calculateManipulatorOnRebirth;
  951. playgroundParticles.pauseCalculationWhenInvisible = pauseCalculationWhenInvisible;
  952. playgroundParticles.calculationTriggerSize = calculationTriggerSize;
  953. playgroundParticles.calculationTriggerOffset = calculationTriggerOffset;
  954. playgroundParticles.calculationTriggerTransform = calculationTriggerTransform;
  955. playgroundParticles.calculationTriggerSizeGizmo = calculationTriggerSizeGizmo;
  956. playgroundParticles.forceVisibilityWhenOutOfFrustrum = forceVisibilityWhenOutOfFrustrum;
  957. playgroundParticles.syncPositionsOnMainThread = syncPositionsOnMainThread;
  958. playgroundParticles.applyLockPosition = applyLockPosition;
  959. playgroundParticles.applyLockRotation = applyLockRotation;
  960. playgroundParticles.applyLockScale = applyLockScale;
  961. playgroundParticles.lockPositionIsLocal = lockPositionIsLocal;
  962. playgroundParticles.lockRotationIsLocal = lockRotationIsLocal;
  963. playgroundParticles.lockPosition = lockPosition;
  964. playgroundParticles.lockRotation = lockRotation;
  965. playgroundParticles.lockScale = lockScale;
  966. playgroundParticles.applyMovementCompensationLifetimeStrength = applyMovementCompensationLifetimeStrength;
  967. playgroundParticles.movementCompensationLifetimeStrength = new AnimationCurve(movementCompensationLifetimeStrength.keys);
  968. playgroundParticles.applyParticleMask = applyParticleMask;
  969. playgroundParticles.particleMask = particleMask;
  970. playgroundParticles.particleMaskTime = particleMaskTime;
  971. playgroundParticles.particleMaskSorting = particleMaskSorting;
  972. playgroundParticles.stretchSpeed = stretchSpeed;
  973. playgroundParticles.applyStretchStartDirection = applyStretchStartDirection;
  974. playgroundParticles.applyLifetimeStretching = applyLifetimeStretching;
  975. playgroundParticles.stretchLifetime = new AnimationCurve(stretchLifetime.keys);
  976. playgroundParticles.threadMethod = threadMethod;
  977. // Scripted source variables
  978. playgroundParticles.scriptedEmissionIndex = scriptedEmissionIndex;
  979. playgroundParticles.scriptedEmissionPosition = scriptedEmissionPosition;
  980. playgroundParticles.scriptedEmissionVelocity = scriptedEmissionVelocity;
  981. playgroundParticles.scriptedEmissionColor = scriptedEmissionColor;
  982. // Collision detection
  983. playgroundParticles.collision = collision;
  984. playgroundParticles.affectRigidbodies = affectRigidbodies;
  985. playgroundParticles.inverseRigidbodyCollision = inverseRigidbodyCollision;
  986. playgroundParticles.mass = mass;
  987. playgroundParticles.collisionRadius = collisionRadius;
  988. playgroundParticles.collisionMask = collisionMask;
  989. playgroundParticles.bounciness = bounciness;
  990. playgroundParticles.lifetimeLoss = lifetimeLoss;
  991. playgroundParticles.bounceRandomMin = bounceRandomMin;
  992. playgroundParticles.bounceRandomMax = bounceRandomMax;
  993. playgroundParticles.collisionType = collisionType;
  994. playgroundParticles.minCollisionDepth = minCollisionDepth;
  995. playgroundParticles.maxCollisionDepth = maxCollisionDepth;
  996. playgroundParticles.stickyCollisions = stickyCollisions;
  997. playgroundParticles.stickyCollisionsSurfaceOffset = stickyCollisionsSurfaceOffset;
  998. playgroundParticles.stickyCollisionsMask = stickyCollisionsMask;
  999. playgroundParticles.collisionPrecision = collisionPrecision;
  1000. playgroundParticles.forceCollisionCaching = forceCollisionCaching;
  1001. playgroundParticles.collisionExclusion = new List<Transform>();
  1002. for (int i = 0; i<collisionExclusion.Count; i++)
  1003. playgroundParticles.collisionExclusion.Add(collisionExclusion[i]);
  1004. playgroundParticles.colliders = new List<PlaygroundColliderC>();
  1005. for (int i = 0; i<colliders.Count; i++)
  1006. playgroundParticles.colliders.Add(colliders[i].Clone());
  1007. // States (source)
  1008. playgroundParticles.states = new List<ParticleStateC>();
  1009. for (int i = 0; i<states.Count; i++) {
  1010. playgroundParticles.states.Add(states[i].Clone());
  1011. }
  1012. // Spline
  1013. playgroundParticles.splines = new List<PlaygroundSpline>();
  1014. for (int i = 0; i<splines.Count; i++)
  1015. playgroundParticles.splines.Add(splines[i]);
  1016. playgroundParticles.splineTimeOffset = splineTimeOffset;
  1017. playgroundParticles.treatAsOneSpline = treatAsOneSpline;
  1018. // Scene objects (source)
  1019. playgroundParticles.worldObject = worldObject.Clone();
  1020. playgroundParticles.skinnedWorldObject = skinnedWorldObject.Clone();
  1021. playgroundParticles.forceSkinnedMeshUpdateOnMainThread = forceSkinnedMeshUpdateOnMainThread;
  1022. playgroundParticles.sourceTransform = sourceTransform;
  1023. playgroundParticles.sourceTransforms = new List<PlaygroundTransformC>();
  1024. for (int i = 0; i<sourceTransforms.Count; i++)
  1025. playgroundParticles.sourceTransforms.Add(sourceTransforms[i].Clone());
  1026. playgroundParticles.treatAsOneTransform = treatAsOneTransform;
  1027. // Paint
  1028. playgroundParticles.paint = paint.Clone();
  1029. // Projection
  1030. playgroundParticles.projection = projection.Clone();
  1031. // Manipulators
  1032. playgroundParticles.manipulators = new List<ManipulatorObjectC>();
  1033. for (int i = 0; i<manipulators.Count; i++)
  1034. playgroundParticles.manipulators.Add(manipulators[i].Clone());
  1035. // Events
  1036. playgroundParticles.events = new List<PlaygroundEventC>();
  1037. for (int i = 0; i<events.Count; i++)
  1038. playgroundParticles.events.Add(events[i].Clone());
  1039. // Lifetime Colors
  1040. playgroundParticles.lifetimeColors = new List<PlaygroundGradientC>();
  1041. for (int i = 0; i<lifetimeColors.Count; i++) {
  1042. playgroundParticles.lifetimeColors.Add(new PlaygroundGradientC());
  1043. lifetimeColors[i].CopyTo(playgroundParticles.lifetimeColors[i]);
  1044. }
  1045. // Turbulence
  1046. playgroundParticles.turbulenceType = turbulenceType;
  1047. playgroundParticles.turbulenceApplyLifetimeStrength = turbulenceApplyLifetimeStrength;
  1048. playgroundParticles.turbulenceLifetimeStrength = new AnimationCurve(turbulenceLifetimeStrength.keys);
  1049. playgroundParticles.turbulenceScale = turbulenceScale;
  1050. playgroundParticles.turbulenceStrength = turbulenceStrength;
  1051. playgroundParticles.turbulenceTimeScale = turbulenceTimeScale;
  1052. // Other
  1053. playgroundParticles.particleTimescale = particleTimescale;
  1054. }
  1055. /// <summary>
  1056. /// Copies stored data of Snapshots into a particle system (separated from CopyTo() to solve Save/Load overwrite paradox).
  1057. /// </summary>
  1058. /// <param name="playgroundParticles">Playground particles.</param>
  1059. public void CopySaveDataTo (PlaygroundParticlesC playgroundParticles) {
  1060. playgroundParticles.snapshots = new List<PlaygroundSave>();
  1061. for (int i = 0; i<snapshots.Count; i++)
  1062. playgroundParticles.snapshots.Add(snapshots[i].Clone());
  1063. }
  1064. /// <summary>
  1065. /// Sets emission On or Off.
  1066. /// </summary>
  1067. /// <param name="setEmission">If set to <c>true</c> then emit particles.</param>
  1068. public void Emit (bool setEmission) {
  1069. emit = setEmission;
  1070. if (emit) {
  1071. simulationStarted = localTime;
  1072. calculate = true;
  1073. hasActiveParticles = true;
  1074. threadHadNoActiveParticles = false;
  1075. loopExceeded = false;
  1076. loopExceededOnParticle = -1;
  1077. if (thisInstance==null)
  1078. thisInstance = this;
  1079. particleSystemGameObject.SetActive(true);
  1080. Emission(thisInstance, true, true);
  1081. } else {
  1082. emissionStopped = localTime;
  1083. if (clearParticlesOnEmissionStop)
  1084. InactivateParticles();
  1085. }
  1086. previousEmission = setEmission;
  1087. }
  1088. /// <summary>
  1089. /// Emits a single particle at previously set scripted emission position, velocity and color.
  1090. /// </summary>
  1091. public int Emit () {
  1092. source = SOURCEC.Script;
  1093. int returnIndex = scriptedEmissionIndex;
  1094. EmitProcedure(scriptedEmissionPosition, scriptedEmissionVelocity, scriptedEmissionColor);
  1095. //particleSystemGameObject.SetActive(true);
  1096. return returnIndex;
  1097. }
  1098. /// <summary>
  1099. /// Emits a single particle at position with previously set scripted emission velocity and color.
  1100. /// </summary>
  1101. /// <param name="givePosition">Position.</param>
  1102. public int Emit (Vector3 givePosition) {
  1103. source = SOURCEC.Script;
  1104. int returnIndex = scriptedEmissionIndex;
  1105. EmitProcedure(givePosition, scriptedEmissionVelocity, scriptedEmissionColor);
  1106. //particleSystemGameObject.SetActive(true);
  1107. return returnIndex;
  1108. }
  1109. /// <summary>
  1110. /// Emits a single particle at position with velocity, the color will be set from the previous scripted emission color.
  1111. /// </summary>
  1112. /// <param name="givePosition">Position.</param>
  1113. /// <param name="giveVelocity">Velocity.</param>
  1114. public int Emit (Vector3 givePosition, Vector3 giveVelocity) {
  1115. source = SOURCEC.Script;
  1116. int returnIndex = scriptedEmissionIndex;
  1117. EmitProcedure(givePosition, giveVelocity, scriptedEmissionColor);
  1118. return returnIndex;
  1119. }
  1120. /// <summary>
  1121. /// Emits a single particle at position with velocity and color (Source Mode SOURCEC.Script will be automatically set).
  1122. /// </summary>
  1123. /// <param name="givePosition">Position.</param>
  1124. /// <param name="giveVelocity">Velocity.</param>
  1125. /// <param name="giveColor">Color.</param>
  1126. public int Emit (Vector3 givePosition, Vector3 giveVelocity, Color32 giveColor) {
  1127. source = SOURCEC.Script;
  1128. int returnIndex = scriptedEmissionIndex;
  1129. EmitProcedure(givePosition, giveVelocity, giveColor);
  1130. return returnIndex;
  1131. }
  1132. /// <summary>
  1133. /// Emits number of particles set by quantity. All other values will be set from the previous scripted emission call (or as set in Inspector).
  1134. /// </summary>
  1135. /// <param name="quantity">Quantity.</param>
  1136. public void Emit (int quantity) {
  1137. source = SOURCEC.Script;
  1138. for (int i = 0; i<quantity; i++)
  1139. EmitProcedure(
  1140. scriptedEmissionPosition,
  1141. scriptedEmissionVelocity,
  1142. scriptedEmissionColor
  1143. );
  1144. }
  1145. /// <summary>
  1146. /// Emits number of particles set by quantity, position and minimum - maximum random velocity.
  1147. /// </summary>
  1148. /// <param name="quantity">Quantity.</param>
  1149. /// <param name="givePosition">Position.</param>
  1150. /// <param name="randomVelocityMin">Random minimum velocity.</param>
  1151. /// <param name="randomVelocityMax">Random maximum velocity.</param>
  1152. /// <param name="giveColor">Color.</param>
  1153. public void Emit (int quantity, Vector3 givePosition, Vector3 randomVelocityMin, Vector3 randomVelocityMax, Color32 giveColor) {
  1154. source = SOURCEC.Script;
  1155. for (int i = 0; i<quantity; i++)
  1156. EmitProcedure(
  1157. givePosition,
  1158. applyInitialVelocityShape?
  1159. Vector3.Scale (RandomRange(internalRandom01, randomVelocityMin, randomVelocityMax), initialVelocityShape.Evaluate((i*1f)/(quantity*1f), initialVelocityShapeScale))
  1160. :
  1161. RandomRange(internalRandom01, randomVelocityMin, randomVelocityMax),
  1162. giveColor
  1163. );
  1164. }
  1165. /// <summary>
  1166. /// Emits number of particles set by quantity, minimum - maximum random position and velocity.
  1167. /// </summary>
  1168. /// <param name="quantity">Quantity.</param>
  1169. /// <param name="randomPositionMin">Random position minimum.</param>
  1170. /// <param name="randomPositionMax">Random position max.</param>
  1171. /// <param name="randomVelocityMin">Random velocity minimum.</param>
  1172. /// <param name="randomVelocityMax">Random velocity max.</param>
  1173. /// <param name="giveColor">Color.</param>
  1174. public void Emit (int quantity, Vector3 randomPositionMin, Vector3 randomPositionMax, Vector3 randomVelocityMin, Vector3 randomVelocityMax, Color32 giveColor) {
  1175. source = SOURCEC.Script;
  1176. for (int i = 0; i<quantity; i++)
  1177. EmitProcedure(
  1178. RandomRange(internalRandom01, randomPositionMin, randomPositionMax),
  1179. applyInitialVelocityShape?
  1180. Vector3.Scale (RandomRange(internalRandom01, randomVelocityMin, randomVelocityMax), initialVelocityShape.Evaluate((i*1f)/(quantity*1f), initialVelocityShapeScale))
  1181. :
  1182. RandomRange(internalRandom01, randomVelocityMin, randomVelocityMax),
  1183. giveColor
  1184. );
  1185. }
  1186. /// <summary>
  1187. /// Emits a single particle with specified lifetime. This will set particle position, velocity and color from previously called emission (as set in the Source tab).
  1188. /// </summary>
  1189. /// <param name="giveLifetime">Lifetime.</param>
  1190. public int Emit (float giveLifetime) {
  1191. source = SOURCEC.Script;
  1192. int returnIndex = scriptedEmissionIndex;
  1193. scriptedLifetime = giveLifetime;
  1194. EmitProcedure(scriptedEmissionPosition, scriptedEmissionVelocity, scriptedEmissionColor);
  1195. scriptedLifetime = 0;
  1196. return returnIndex;
  1197. }
  1198. /// <summary>
  1199. /// Emits a single particle with specified lifetime. This will set particle velocity and color from previously called emission (as set in the Source tab).
  1200. /// </summary>
  1201. /// <param name="givePosition">Position.</param>
  1202. /// <param name="giveLifetime">Lifetime.</param>
  1203. public int Emit (Vector3 givePosition, float giveLifetime) {
  1204. source = SOURCEC.Script;
  1205. int returnIndex = scriptedEmissionIndex;
  1206. scriptedLifetime = giveLifetime;
  1207. EmitProcedure(givePosition, scriptedEmissionVelocity, scriptedEmissionColor);
  1208. scriptedLifetime = 0;
  1209. return returnIndex;
  1210. }
  1211. /// <summary>
  1212. /// Emits a single particle with specified lifetime. This will set particle color from previously called emission (as set in the Source tab).
  1213. /// </summary>
  1214. /// <param name="givePosition">Position.</param>
  1215. /// <param name="giveVelocity">Velocity.</param>
  1216. /// <param name="giveLifetime">Lifetime.</param>
  1217. public int Emit (Vector3 givePosition, Vector3 giveVelocity, float giveLifetime) {
  1218. source = SOURCEC.Script;
  1219. int returnIndex = scriptedEmissionIndex;
  1220. scriptedLifetime = giveLifetime;
  1221. EmitProcedure(givePosition, giveVelocity, scriptedEmissionColor);
  1222. scriptedLifetime = 0;
  1223. return returnIndex;
  1224. }
  1225. /// <summary>
  1226. /// Emits a single particle with specified position, velocity, lifetime and color.
  1227. /// </summary>
  1228. /// <param name="givePosition">Position.</param>
  1229. /// <param name="giveVelocity">Velocity.</param>
  1230. /// <param name="giveLifetime">Lifetime.</param>
  1231. /// <param name="giveColor">Color.</param>
  1232. public int Emit (Vector3 givePosition, Vector3 giveVelocity, float giveLifetime, Color32 giveColor) {
  1233. source = SOURCEC.Script;
  1234. int returnIndex = scriptedEmissionIndex;
  1235. scriptedLifetime = giveLifetime;
  1236. EmitProcedure(givePosition, giveVelocity, giveColor);
  1237. scriptedLifetime = 0;
  1238. return returnIndex;
  1239. }
  1240. /// <summary>
  1241. /// Thread-safe version of Emit().
  1242. /// </summary>
  1243. /// <param name="givePosition">Position.</param>
  1244. /// <param name="giveVelocity">Velocity.</param>
  1245. /// <param name="giveColor">Color.</param>
  1246. public void ThreadSafeEmit (Vector3 givePosition, Vector3 giveVelocity, Color32 giveColor) {
  1247. EmitProcedure(givePosition, giveVelocity, giveColor);
  1248. }
  1249. public void ThreadSafeEmit (int quantity, Vector3 givePosition, Vector3 randomVelocityMin, Vector3 randomVelocityMax, Color32 giveColor) {
  1250. for (int i = 0; i<quantity; i++)
  1251. EmitProcedure(
  1252. givePosition,
  1253. applyInitialVelocityShape?
  1254. Vector3.Scale (RandomRange(internalRandom01, randomVelocityMin, randomVelocityMax), initialVelocityShape.Evaluate((i*1f)/(quantity*1f), initialVelocityShapeScale))
  1255. :
  1256. RandomRange(internalRandom01, randomVelocityMin, randomVelocityMax),
  1257. giveColor
  1258. );
  1259. }
  1260. /// <summary>
  1261. /// Internal emission procedure called upon Emit().
  1262. /// </summary>
  1263. /// <param name="givePosition">Give position.</param>
  1264. /// <param name="giveVelocity">Give velocity.</param>
  1265. /// <param name="giveColor">Give color.</param>
  1266. void EmitProcedure (Vector3 givePosition, Vector3 giveVelocity, Color32 giveColor) {
  1267. scriptedEmissionIndex=scriptedEmissionIndex%particleCount;
  1268. scriptedEmissionPosition = givePosition;
  1269. scriptedEmissionVelocity = giveVelocity;
  1270. scriptedEmissionColor = giveColor;
  1271. hasActiveParticles = true;
  1272. threadHadNoActiveParticles = false;
  1273. cameFromNonCalculatedFrame = false;
  1274. cameFromNonEmissionFrame = false;
  1275. playgroundCache.simulate[scriptedEmissionIndex] = true;
  1276. Rebirth(thisInstance, scriptedEmissionIndex, internalRandom01);
  1277. if (playgroundCache.lifetimeOffset.Length!=particleCount) return;
  1278. playgroundCache.initialColor[scriptedEmissionIndex] = scriptedEmissionColor;
  1279. playgroundCache.lifetimeOffset[scriptedEmissionIndex] = 0;
  1280. playgroundCache.life[scriptedEmissionIndex] = 0;
  1281. playgroundCache.birth[scriptedEmissionIndex] = PlaygroundC.globalTime;
  1282. if (scriptedLifetime==0) {
  1283. playgroundCache.death[scriptedEmissionIndex] = playgroundCache.birth[scriptedEmissionIndex]+lifetime;
  1284. } else {
  1285. playgroundCache.death[scriptedEmissionIndex] = playgroundCache.birth[scriptedEmissionIndex]+scriptedLifetime;
  1286. }
  1287. playgroundCache.emission[scriptedEmissionIndex] = true;
  1288. playgroundCache.scriptedColor[scriptedEmissionIndex] = new Color(giveColor.r, giveColor.g, giveColor.b, giveColor.a);
  1289. playgroundCache.isFirstLoop[scriptedEmissionIndex] = true;
  1290. emit = true;
  1291. previousEmission = true;
  1292. simulationStarted = localTime;
  1293. loopExceeded = false;
  1294. loopExceededOnParticle = -1;
  1295. scriptedEmissionIndex++;scriptedEmissionIndex=scriptedEmissionIndex%particleCount;
  1296. }
  1297. /// <summary>
  1298. /// Checks if particles are still in simulation.
  1299. /// </summary>
  1300. /// <returns><c>true</c> if this particle system is alive; otherwise, <c>false</c>.</returns>
  1301. public bool IsAlive () {
  1302. return calculate && hasActiveParticles;
  1303. }
  1304. /// <summary>
  1305. /// Determines whether this particle system is simulated in local space.
  1306. /// </summary>
  1307. /// <returns><c>true</c> if this particle system is simulated in local space; otherwise, <c>false</c>.</returns>
  1308. public bool IsLocalSpace () {
  1309. return localSpace;
  1310. }
  1311. /// <summary>
  1312. /// Determines if this particle system is in a transition.
  1313. /// </summary>
  1314. /// <returns><c>true</c>, if transition is active, <c>false</c> otherwise.</returns>
  1315. public bool InTransition () {
  1316. return inTransition;
  1317. }
  1318. /// <summary>
  1319. /// Determines whether this particle system is loading a snapshot.
  1320. /// </summary>
  1321. /// <returns><c>true</c> if this particle system is loading a snapshot; otherwise, <c>false</c>.</returns>
  1322. public bool IsLoading () {
  1323. return isLoading;
  1324. }
  1325. /// <summary>
  1326. /// Determines whether this particle system is saving a snapshot.
  1327. /// </summary>
  1328. /// <returns><c>true</c> if this particle system is saving a snapshot; otherwise, <c>false</c>.</returns>
  1329. public bool IsSaving () {
  1330. return isSaving;
  1331. }
  1332. /// <summary>
  1333. /// Check if the particle system is ready. You should wait for IsReady() to become true before calling any scripted emission. This returns the same result as calling Initialized().
  1334. /// </summary>
  1335. /// <returns><c>true</c> if the particle system is ready; otherwise, <c>false</c>.</returns>
  1336. public bool IsReady () {
  1337. return initialized&&!isPrewarming;
  1338. }
  1339. /// <summary>
  1340. /// Check if the particle system is ready. You should wait for Initialized() to become true before calling any scripted emission.
  1341. /// </summary>
  1342. /// <returns><c>true</c> if the particle system is ready; otherwise, <c>false</c>.</returns>
  1343. public bool Initialized () {
  1344. return initialized&&!isPrewarming;
  1345. }
  1346. /// <summary>
  1347. /// Determines whether this particle system is yield refreshing.
  1348. /// </summary>
  1349. /// <returns><c>true</c> if this particle system is yield refreshing; otherwise, <c>false</c>.</returns>
  1350. public bool IsYieldRefreshing () {
  1351. return isYieldRefreshing;
  1352. }
  1353. public bool IsSettingParticleTime () {
  1354. return isSettingParticleTime;
  1355. }
  1356. public bool IsDoneThread {
  1357. get {return isDoneThread;}
  1358. set {isDoneThread = value;}
  1359. }
  1360. /// <summary>
  1361. /// Determines whether the skinned world object is ready.
  1362. /// </summary>
  1363. /// <returns><c>true</c> if this skinned world object is ready; otherwise, <c>false</c>.</returns>
  1364. public bool IsSkinnedWorldObjectReady () {
  1365. return source==SOURCEC.SkinnedWorldObject&&skinnedWorldObjectReady;
  1366. }
  1367. /// <summary>
  1368. /// Determines whether this particle system has turbulence active.
  1369. /// </summary>
  1370. /// <returns><c>true</c> if this particle system has turbulence; otherwise, <c>false</c>.</returns>
  1371. public bool HasTurbulence () {
  1372. return calculate && !onlySourcePositioning && !onlyLifetimePositioning && turbulenceStrength>0 && turbulenceType!=TURBULENCETYPE.None;
  1373. }
  1374. /// <summary>
  1375. /// Determines whether this particle system has overflow set by Overflow Offset.
  1376. /// </summary>
  1377. /// <returns><c>true</c> if this particle system has overflow; otherwise, <c>false</c>.</returns>
  1378. public bool HasOverflow () {
  1379. return overflow;
  1380. }
  1381. /// <summary>
  1382. /// Determines whether this particle system may be affected by one or more available Global Manipulators.
  1383. /// </summary>
  1384. /// <returns><c>true</c> if this particle system will be affected by one or more Global Manipulators; otherwise, <c>false</c>.</returns>
  1385. public bool HasGlobalManipulator () {
  1386. return hasGlobalAffectingManipulators;
  1387. }
  1388. /// <summary>
  1389. /// Determines whether this particle system has an initialized collision cache.
  1390. /// </summary>
  1391. /// <returns><c>true</c> if this particle system has a collision cache; otherwise, <c>false</c>.</returns>
  1392. public bool HasCollisionCache () {
  1393. return hasCollisionCache;
  1394. }
  1395. /// <summary>
  1396. /// Determines if a particle at specified index has collided during its lifetime.
  1397. /// </summary>
  1398. /// <returns><c>true</c> if the particle at index has collided during its lifetime; otherwise, <c>false</c>.</returns>
  1399. /// <param name="index">Index.</param>
  1400. public bool HasCollided (int index) {
  1401. if (index<0||index>particleCount-1)
  1402. return false;
  1403. if (hasCollisionCache)
  1404. return collisionCache.hasCollided[index];
  1405. return false;
  1406. }
  1407. /// <summary>
  1408. /// Gets the particle collision position at index.
  1409. /// </summary>
  1410. /// <returns>The particle collision position.</returns>
  1411. /// <param name="index">Index.</param>
  1412. public Vector3 GetCollisionPosition (int index) {
  1413. if (!hasCollisionCache || index<0||index>particleCount-1)
  1414. return Vector3.zero;
  1415. return collisionCache.collisionPosition[index];
  1416. }
  1417. /// <summary>
  1418. /// Gets the particle collision normal at index.
  1419. /// </summary>
  1420. /// <returns>The particle collision normal.</returns>
  1421. /// <param name="index">Index.</param>
  1422. public Vector3 GetCollisionNormal (int index) {
  1423. if (!hasCollisionCache || index<0||index>particleCount-1)
  1424. return Vector3.zero;
  1425. return collisionCache.collisionNormal[index];
  1426. }
  1427. /// <summary>
  1428. /// Gets the particle sticky position at index. Note that stickyCollisions must be enabled to return expected values.
  1429. /// </summary>
  1430. /// <returns>The particle sticky position.</returns>
  1431. /// <param name="index">Index.</param>
  1432. public Vector3 GetStickyPosition (int index) {
  1433. if (!hasCollisionCache || index<0||index>particleCount-1)
  1434. return Vector3.zero;
  1435. return collisionCache.stickyPosition[index];
  1436. }
  1437. /// <summary>
  1438. /// Sets the particle at index to stick onto a parent transform.
  1439. /// </summary>
  1440. /// <param name="index">Index.</param>
  1441. /// <param name="position">Position.</param>
  1442. /// <param name="normal">Normal.</param>
  1443. /// <param name="parent">Parent.</param>
  1444. public void SetSticky (int index, Vector3 position, Vector3 normal, float offset, Transform parent) {
  1445. if (index<0||index>particleCount-1)
  1446. return;
  1447. if (!hasCollisionCache) {
  1448. collisionCache = new CollisionCache(particleCount);
  1449. hasCollisionCache = true;
  1450. }
  1451. collisionCache.SetSticky(index, position, normal, stickyCollisionsSurfaceOffset, parent);
  1452. }
  1453. /// <summary>
  1454. /// Updates the sticky particle position.
  1455. /// </summary>
  1456. /// <param name="index">Index.</param>
  1457. public void UpdateSticky (int index) {
  1458. if (!hasCollisionCache || index<0||index>particleCount-1)
  1459. return;
  1460. collisionCache.hasCollided[index] = true;
  1461. collisionCache.UpdateStickyPosition(index);
  1462. playgroundCache.position[index] = collisionCache.stickyPosition[index];
  1463. particleCache[index].position = playgroundCache.position[index];
  1464. }
  1465. /// <summary>
  1466. /// Clears the collisions. Use this if you for example want to toggle between sticky and non-sticky particle behaviors.
  1467. /// </summary>
  1468. public void ClearCollisions () {
  1469. if (!hasCollisionCache)
  1470. return;
  1471. collisionCache.ClearCollisions();
  1472. }
  1473. /// <summary>
  1474. /// Clears the collisions at index. Use this if you for example want to toggle between sticky and non-sticky particle behaviors.
  1475. /// </summary>
  1476. public void ClearCollisions (int index) {
  1477. if (!hasCollisionCache || index<0||index>particleCount-1)
  1478. return;
  1479. collisionCache.Reset(index);
  1480. }
  1481. /// <summary>
  1482. /// Gets the particle collision transform at index.
  1483. /// </summary>
  1484. /// <returns>The particle collision transform.</returns>
  1485. /// <param name="index">Index.</param>
  1486. public Transform GetCollisionTransform (int index) {
  1487. if (!hasCollisionCache || index<0||index>particleCount-1)
  1488. return null;
  1489. return collisionCache.collisionTransform[index];
  1490. }
  1491. /// <summary>
  1492. /// Gets the layer this particle system is within. This is safe to call from another thread.
  1493. /// </summary>
  1494. /// <returns>The layer.</returns>
  1495. public int GetLayer () {
  1496. return thisLayer;
  1497. }
  1498. /// <summary>
  1499. /// Sets the random seed for the internal System.Random.
  1500. /// </summary>
  1501. /// <param name="seed">Seed.</param>
  1502. public void SetRandomSeed (int seed) {
  1503. internalRandom01 = new System.Random(seed);
  1504. }
  1505. int reportUpdateStepper = 100;
  1506. bool isReportingBadUpdateRate = false;
  1507. /// <summary>
  1508. /// Determines whether this particle system is reporting bad update rate. You will also see the "P" icon indicator in the Hierarchy turn red in Play Mode whenever this returns true.
  1509. /// </summary>
  1510. /// <returns><c>true</c> if this particle system is reporting bad update rate; otherwise, <c>false</c>.</returns>
  1511. public bool IsReportingBadUpdateRate () {
  1512. if (isPrewarming||inTransition) return false;
  1513. reportUpdateStepper--;
  1514. if (reportUpdateStepper==0) {
  1515. reportUpdateStepper = 100;
  1516. isReportingBadUpdateRate = localTime>1f && localDeltaTime>.04f;
  1517. }
  1518. return isReportingBadUpdateRate;
  1519. }
  1520. /// <summary>
  1521. /// Gets the current delta time of a particle system's update loop.
  1522. /// </summary>
  1523. /// <returns>The delta time.</returns>
  1524. public float GetDeltaTime () {
  1525. return t;
  1526. }
  1527. public float LastTimeUpdated {
  1528. get {return lastTimeUpdated;}
  1529. set {lastTimeUpdated = value;}
  1530. }
  1531. /// <summary>
  1532. /// Gets the running simplex algorithm.
  1533. /// </summary>
  1534. /// <returns>The running simplex algorithm.</returns>
  1535. public SimplexNoise GetSimplex () {
  1536. if (turbulenceSimplex==null)
  1537. turbulenceSimplex = new SimplexNoise();
  1538. return turbulenceSimplex;
  1539. }
  1540. /// <summary>
  1541. /// Kill the specified particle.
  1542. /// </summary>
  1543. /// <param name="p">Index of particle.</param>
  1544. public void Kill (int p) {
  1545. if (p>=particleCount || p<0) return;
  1546. playgroundCache.changedByPropertyDeath[p] = true;
  1547. playgroundCache.life[p] = lifetime;
  1548. playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  1549. particleCache[p].position = playgroundCache.position[p];
  1550. playgroundCache.manipulatorId[p] = 0;
  1551. }
  1552. /// <summary>
  1553. /// Kill the specified particle and send death events to all Manipulators tracking particles.
  1554. /// </summary>
  1555. /// <param name="p">Index of particle.</param>
  1556. public void KillAndSendManipulatorDeathEvents (int p) {
  1557. if (p>=particleCount || p<0) return;
  1558. playgroundCache.changedByPropertyDeath[p] = true;
  1559. playgroundCache.life[p] = lifetime;
  1560. playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  1561. particleCache[p].position = playgroundCache.position[p];
  1562. // Check if particle has a connection to a local manipulator to send death event
  1563. if (playgroundCache.manipulatorId[p]!=0) {
  1564. for (int m = 0; m<manipulators.Count; m++) {
  1565. if (manipulators[m].trackParticles &&
  1566. manipulators[m].ContainsParticle(particleSystemId, p)) {
  1567. playgroundCache.manipulatorId[p] = 0;
  1568. manipulators[m].RemoveParticle(particleSystemId, p);
  1569. if (manipulators[m].sendEventExit) {
  1570. UpdateEventParticle(manipulators[m].manipulatorEventParticle, p);
  1571. manipulators[m].SendParticleEventDeath();
  1572. }
  1573. }
  1574. }
  1575. // Check if particle has a connection to a global manipulator to send death event
  1576. if (playgroundCache.manipulatorId[p]!=0) {
  1577. for (int m = 0; m<PlaygroundC.reference.manipulators.Count; m++) {
  1578. if (PlaygroundC.reference.manipulators[m].trackParticles &&
  1579. PlaygroundC.reference.manipulators[m].ContainsParticle(particleSystemId, p)) {
  1580. PlaygroundC.reference.manipulators[m].RemoveParticle(particleSystemId, p);
  1581. if (PlaygroundC.reference.manipulators[m].sendEventExit) {
  1582. UpdateEventParticle(PlaygroundC.reference.manipulators[m].manipulatorEventParticle, p);
  1583. PlaygroundC.reference.manipulators[m].SendParticleEventDeath();
  1584. }
  1585. }
  1586. }
  1587. }
  1588. playgroundCache.manipulatorId[p] = 0;
  1589. }
  1590. }
  1591. /// <summary>
  1592. /// Sets a particle to no longer respond to forces.
  1593. /// </summary>
  1594. /// <param name="p">Index of particle.</param>
  1595. /// <param name="noForce">If set to <c>true</c> then disable force for the particle at index.</param>
  1596. public void SetNoForce (int p, bool noForce) {
  1597. playgroundCache.noForce[p] = noForce;
  1598. }
  1599. /// <summary>
  1600. /// Determines if the particle have noForce set to true
  1601. /// </summary>
  1602. /// <returns><c>true</c>, if force is disabled, <c>false</c> otherwise.</returns>
  1603. /// <param name="p">Index of particle.</param>
  1604. public bool NoForce (int p) {
  1605. return playgroundCache.noForce[p];
  1606. }
  1607. /// <summary>
  1608. /// Translate the specified particle
  1609. /// </summary>
  1610. /// <param name="p">Particle index.</param>
  1611. /// <param name="translation">Translation.</param>
  1612. public void Translate (int p, Vector3 translation) {
  1613. playgroundCache.position[p] += translation;
  1614. }
  1615. /// <summary>
  1616. /// Positions the specified particle.
  1617. /// </summary>
  1618. /// <param name="p">Particle index.</param>
  1619. /// <param name="position">Position.</param>
  1620. public void Position (int p, Vector3 position) {
  1621. playgroundCache.position[p] = position;
  1622. }
  1623. /// <summary>
  1624. /// Positions to transform point.
  1625. /// </summary>
  1626. /// <param name="p">Particle index.</param>
  1627. /// <param name="position">Position.</param>
  1628. /// <param name="targetTransform">Target transform.</param>
  1629. public void PositionToTransformPoint (int p, Vector3 position, Transform targetTransform) {
  1630. playgroundCache.position[p] = targetTransform.TransformPoint (position);
  1631. }
  1632. /// <summary>
  1633. /// Positions to inverse transform point.
  1634. /// </summary>
  1635. /// <param name="p">Particle index.</param>
  1636. /// <param name="position">Position.</param>
  1637. /// <param name="targetTransform">Target transform.</param>
  1638. public void PositionToInverseTransformPoint (int p, Vector3 position, Transform targetTransform) {
  1639. playgroundCache.position[p] = targetTransform.InverseTransformPoint (position);
  1640. }
  1641. /// <summary>
  1642. /// Gets the particle position.
  1643. /// </summary>
  1644. /// <returns>The particle position.</returns>
  1645. /// <param name="p">Particle index.</param>
  1646. public Vector3 GetParticlePosition (int p) {
  1647. return playgroundCache.position[p];
  1648. }
  1649. public void SetHasActiveParticles () {
  1650. hasActiveParticles = true;
  1651. threadHadNoActiveParticles = false;
  1652. }
  1653. /// <summary>
  1654. /// Determines whether this particle system has several manipulator events.
  1655. /// </summary>
  1656. /// <returns><c>true</c> if this particle system has several manipulator events; otherwise, <c>false</c>.</returns>
  1657. public bool HasSeveralManipulatorEvents () {
  1658. return hasSeveralManipulatorEvents;
  1659. }
  1660. /// <summary>
  1661. /// Protects the particle from specified manipulator.
  1662. /// </summary>
  1663. /// <param name="particle">Particle index.</param>
  1664. /// <param name="manipulator">Manipulator.</param>
  1665. public void ProtectParticleFromManipulator (int particle, ManipulatorObjectC manipulator) {
  1666. if (manipulator.transform.Update())
  1667. playgroundCache.excludeFromManipulatorId[particle] = manipulator.transform.instanceID;
  1668. }
  1669. /// <summary>
  1670. /// Removes the particle protection.
  1671. /// </summary>
  1672. /// <param name="particle">Particle index.</param>
  1673. public void RemoveParticleProtection (int particle) {
  1674. playgroundCache.excludeFromManipulatorId[particle] = 0;
  1675. }
  1676. /// <summary>
  1677. /// Determines whether this particle system is currently setting the lifetime.
  1678. /// As the particle system may run on a second thread this can be used to wait until it is ready to simulate.
  1679. /// </summary>
  1680. /// <returns><c>true</c> if this particle system is setting the lifetime; otherwise, <c>false</c>.</returns>
  1681. public bool IsSettingLifetime () {
  1682. return isSettingLifetime;
  1683. }
  1684. /// <summary>
  1685. /// Determines whether this particle system is currently prewarming.
  1686. /// </summary>
  1687. /// <returns><c>true</c> if this particle system is prewarming; otherwise, <c>false</c>.</returns>
  1688. public bool IsPrewarming () {
  1689. return isPrewarming;
  1690. }
  1691. /// <summary>
  1692. /// Determines whether a particle is inside the area of a Manipulator.
  1693. /// </summary>
  1694. /// <returns><c>true</c> if this particle is inside manipulator; otherwise, <c>false</c>.</returns>
  1695. public bool IsParticleInsideManipulator (int particleId, ManipulatorObjectC manipulator) {
  1696. return (manipulator.Contains (playgroundCache.position[particleId], (localSpace?manipulator.transform.localPosition:manipulator.transform.position)));
  1697. }
  1698. /// <summary>
  1699. /// Gets the index of the source spline which currently is read in simulation.
  1700. /// </summary>
  1701. /// <returns>The spline index.</returns>
  1702. public int GetSplineIndex () {
  1703. return splineIndex;
  1704. }
  1705. /// <summary>
  1706. /// Gets the index of the source transform which currently is read in simulation.
  1707. /// </summary>
  1708. /// <returns>The transform index.</returns>
  1709. public int GetTransformIndex () {
  1710. return transformIndex;
  1711. }
  1712. /// <summary>
  1713. /// Refreshes the mask sorting. If you're using a particleMaskSorting of MASKSORTINGC.Scrambled you can use this to randomize new mask positions.
  1714. /// </summary>
  1715. public void RefreshMaskSorting () {
  1716. playgroundCache.maskSorting = new int[particleCount];
  1717. switch (particleMaskSorting) {
  1718. case MASKSORTINGC.Linear:
  1719. for (int i = 0; i<playgroundCache.maskSorting.Length; i++)
  1720. playgroundCache.maskSorting[i] = i;
  1721. break;
  1722. case MASKSORTINGC.Reversed:
  1723. for (int i = 0; i<playgroundCache.maskSorting.Length; i++)
  1724. playgroundCache.maskSorting[i] = (playgroundCache.maskSorting.Length-1)-i;
  1725. break;
  1726. case MASKSORTINGC.Scrambled:
  1727. for (int i = 0; i<playgroundCache.maskSorting.Length; i++)
  1728. playgroundCache.maskSorting[i] = playgroundCache.maskSorting[i] = i;
  1729. PlaygroundC.ShuffleArray(playgroundCache.maskSorting);
  1730. break;
  1731. }
  1732. previousMaskSorting = particleMaskSorting;
  1733. }
  1734. /// <summary>
  1735. /// Creates a new PlaygroundParticlesC object.
  1736. /// </summary>
  1737. /// <returns>The playground particles.</returns>
  1738. /// <param name="images">Images.</param>
  1739. /// <param name="name">Name.</param>
  1740. /// <param name="position">Position.</param>
  1741. /// <param name="rotation">Rotation.</param>
  1742. /// <param name="offset">Offset.</param>
  1743. /// <param name="particleSize">Particle size.</param>
  1744. /// <param name="scale">Scale.</param>
  1745. /// <param name="material">Material.</param>
  1746. public static PlaygroundParticlesC CreatePlaygroundParticles (Texture2D[] images, string name, Vector3 position, Quaternion rotation, Vector3 offset, float particleSize, float scale, Material material) {
  1747. PlaygroundParticlesC playgroundParticles = CreateParticleObject(name,position,rotation,particleSize,material);
  1748. int[] quantityList = new int[images.Length];
  1749. int i = 0;
  1750. for (; i<images.Length; i++)
  1751. quantityList[i] = images[i].width*images[i].height;
  1752. playgroundParticles.particleCache = new ParticleSystem.Particle[quantityList[PlaygroundC.Largest(quantityList)]];
  1753. OnCreatePlaygroundParticles(playgroundParticles);
  1754. for (i = 0; i<images.Length; i++) {
  1755. playgroundParticles.states.Add(new ParticleStateC());
  1756. playgroundParticles.states[playgroundParticles.states.Count-1].ConstructParticles(images[i],scale,offset,"State 0",null);
  1757. }
  1758. return playgroundParticles;
  1759. }
  1760. /// <summary>
  1761. /// Sets default settings for a PlaygroundParticlesC object.
  1762. /// </summary>
  1763. /// <param name="playgroundParticles">Playground particles.</param>
  1764. public static void OnCreatePlaygroundParticles (PlaygroundParticlesC playgroundParticles) {
  1765. playgroundParticles.playgroundCache = new PlaygroundCache();
  1766. playgroundParticles.paint = new PaintObjectC();
  1767. playgroundParticles.states = new List<ParticleStateC>();
  1768. playgroundParticles.projection = new ParticleProjectionC();
  1769. playgroundParticles.colliders = new List<PlaygroundColliderC>();
  1770. playgroundParticles.particleSystemId = PlaygroundC.particlesQuantity-1;
  1771. playgroundParticles.projection.projectionTransform = playgroundParticles.particleSystemTransform;
  1772. playgroundParticles.playgroundCache.initialSize = new float[playgroundParticles.particleCount];
  1773. playgroundParticles.playgroundCache.initialSize = RandomFloat(playgroundParticles.playgroundCache.initialSize.Length, playgroundParticles.sizeMin, playgroundParticles.sizeMax, playgroundParticles.internalRandom01);
  1774. playgroundParticles.previousParticleCount = playgroundParticles.particleCount;
  1775. playgroundParticles.lifetimeSize = new AnimationCurve(new Keyframe(0,1), new Keyframe(1,1));
  1776. playgroundParticles.shurikenParticleSystem.Emit(playgroundParticles.particleCount);
  1777. playgroundParticles.shurikenParticleSystem.GetParticles(playgroundParticles.particleCache);
  1778. for (int p = 0; p<playgroundParticles.particleCache.Length; p++) {
  1779. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p];
  1780. }
  1781. PlaygroundParticlesC.SetParticleCount(playgroundParticles, playgroundParticles.particleCount);
  1782. if (PlaygroundC.reference!=null) {
  1783. PlaygroundC.particlesQuantity++;
  1784. PlaygroundC.reference.particleSystems.Add(playgroundParticles);
  1785. playgroundParticles.particleSystemId = PlaygroundC.particlesQuantity-1;
  1786. }
  1787. }
  1788. /// <summary>
  1789. /// Creates a Shuriken Particle System.
  1790. /// </summary>
  1791. /// <returns>The particle object.</returns>
  1792. /// <param name="name">Name.</param>
  1793. /// <param name="position">Position.</param>
  1794. /// <param name="rotation">Rotation.</param>
  1795. /// <param name="particleSize">Particle size.</param>
  1796. /// <param name="material">Material.</param>
  1797. public static PlaygroundParticlesC CreateParticleObject (string name, Vector3 position, Quaternion rotation, float particleSize, Material material) {
  1798. GameObject go = PlaygroundC.ResourceInstantiate("Particle Playground System");
  1799. PlaygroundParticlesC playgroundParticles = go.GetComponent<PlaygroundParticlesC>();
  1800. playgroundParticles.particleSystemGameObject = go;
  1801. playgroundParticles.particleSystemGameObject.name = name;
  1802. playgroundParticles.shurikenParticleSystem = playgroundParticles.particleSystemGameObject.GetComponent<ParticleSystem>();
  1803. playgroundParticles.particleSystemRenderer = playgroundParticles.shurikenParticleSystem.GetComponent<Renderer>();
  1804. playgroundParticles.particleSystemRenderer2 = playgroundParticles.shurikenParticleSystem.GetComponent<Renderer>() as ParticleSystemRenderer;
  1805. playgroundParticles.particleSystemTransform = playgroundParticles.particleSystemGameObject.transform;
  1806. playgroundParticles.sourceTransform = playgroundParticles.particleSystemTransform;
  1807. playgroundParticles.source = SOURCEC.Transform;
  1808. playgroundParticles.particleSystemTransform.position = position;
  1809. playgroundParticles.particleSystemTransform.rotation = rotation;
  1810. if (PlaygroundC.reference.autoGroup && playgroundParticles.particleSystemTransform.parent==null)
  1811. playgroundParticles.particleSystemTransform.parent = PlaygroundC.referenceTransform;
  1812. if (playgroundParticles.particleSystemRenderer.sharedMaterial==null)
  1813. playgroundParticles.particleSystemRenderer.sharedMaterial = material;
  1814. return playgroundParticles;
  1815. }
  1816. /// <summary>
  1817. /// Creates a new WorldObject.
  1818. /// </summary>
  1819. /// <returns>The world object.</returns>
  1820. /// <param name="meshTransform">Mesh transform.</param>
  1821. public static WorldObject NewWorldObject (Transform meshTransform) {
  1822. WorldObject worldObject = new WorldObject();
  1823. if (meshTransform.GetComponentInChildren<MeshFilter>()) {
  1824. worldObject.transform = meshTransform;
  1825. worldObject.Initialize ();
  1826. } else Debug.Log("Could not find a mesh in "+meshTransform.name+".");
  1827. return worldObject;
  1828. }
  1829. /// <summary>
  1830. /// Creates a new SkinnedWorldObject.
  1831. /// </summary>
  1832. /// <returns>The skinned world object.</returns>
  1833. /// <param name="meshTransform">Mesh transform.</param>
  1834. public static SkinnedWorldObject NewSkinnedWorldObject (Transform meshTransform) {
  1835. SkinnedWorldObject skinnedWorldObject = new SkinnedWorldObject();
  1836. if (meshTransform.GetComponentInChildren<SkinnedMeshRenderer>()) {
  1837. skinnedWorldObject.transform = meshTransform;
  1838. skinnedWorldObject.Initialize ();
  1839. } else Debug.Log("Could not find a skinned mesh in "+meshTransform.name+".");
  1840. return skinnedWorldObject;
  1841. }
  1842. /// <summary>
  1843. /// Creates a new SkinnedWorldObject with pre-set down resolution.
  1844. /// </summary>
  1845. /// <returns>The skinned world object.</returns>
  1846. /// <param name="meshTransform">Mesh transform.</param>
  1847. /// <param name="downResolution">Down resolution.</param>
  1848. public static SkinnedWorldObject NewSkinnedWorldObject (Transform meshTransform, int downResolution) {
  1849. SkinnedWorldObject skinnedWorldObject = NewSkinnedWorldObject(meshTransform);
  1850. skinnedWorldObject.downResolution = downResolution;
  1851. return skinnedWorldObject;
  1852. }
  1853. /// <summary>
  1854. /// Creates a new PaintObject.
  1855. /// </summary>
  1856. /// <returns>The paint object.</returns>
  1857. /// <param name="playgroundParticles">Playground particles.</param>
  1858. public static PaintObjectC NewPaintObject (PlaygroundParticlesC playgroundParticles) {
  1859. PaintObjectC paintObject = new PaintObjectC();
  1860. playgroundParticles.paint = paintObject;
  1861. playgroundParticles.paint.Initialize();
  1862. return paintObject;
  1863. }
  1864. /// <summary>
  1865. /// Creates a new ParticleProjection object.
  1866. /// </summary>
  1867. /// <returns>The projection object.</returns>
  1868. /// <param name="playgroundParticles">Playground particles.</param>
  1869. public static ParticleProjectionC NewProjectionObject (PlaygroundParticlesC playgroundParticles) {
  1870. ParticleProjectionC projectionObject = new ParticleProjectionC();
  1871. playgroundParticles.projection = projectionObject;
  1872. playgroundParticles.projection.Initialize();
  1873. return projectionObject;
  1874. }
  1875. /// <summary>
  1876. /// Creates a new ManipulatorObject and attach to the Playground Manager.
  1877. /// </summary>
  1878. /// <returns>The manipulator object.</returns>
  1879. /// <param name="type">Type.</param>
  1880. /// <param name="affects">Affects.</param>
  1881. /// <param name="manipulatorTransform">Manipulator transform.</param>
  1882. /// <param name="size">Size.</param>
  1883. /// <param name="strength">Strength.</param>
  1884. /// <param name="playgroundParticles">Playground particles.</param>
  1885. public static ManipulatorObjectC NewManipulatorObject (MANIPULATORTYPEC type, LayerMask affects, Transform manipulatorTransform, float size, float strength, PlaygroundParticlesC playgroundParticles) {
  1886. ManipulatorObjectC manipulatorObject = new ManipulatorObjectC();
  1887. manipulatorObject.type = type;
  1888. manipulatorObject.affects = affects;
  1889. manipulatorObject.transform.transform = manipulatorTransform;
  1890. manipulatorObject.size = size;
  1891. manipulatorObject.strength = strength;
  1892. manipulatorObject.bounds = new Bounds(Vector3.zero, new Vector3(size, size, size));
  1893. manipulatorObject.property = new ManipulatorPropertyC();
  1894. manipulatorObject.Update();
  1895. // Add this to Playground Manager or the passed in playgroundParticles
  1896. if (playgroundParticles==null)
  1897. PlaygroundC.reference.manipulators.Add(manipulatorObject);
  1898. else
  1899. playgroundParticles.manipulators.Add(manipulatorObject);
  1900. return manipulatorObject;
  1901. }
  1902. // Set color from PixelParticle object
  1903. public static void SetColor (PlaygroundParticlesC playgroundParticles, int to) {
  1904. Color color = new Color();
  1905. for (int i = 0; i<playgroundParticles.particleCache.Length; i++) {
  1906. color = playgroundParticles.states[to].GetColor(i%playgroundParticles.states[to].colorLength);
  1907. playgroundParticles.particleCache[i].color = color;
  1908. }
  1909. }
  1910. // Set color from Color
  1911. public static void SetColor (PlaygroundParticlesC playgroundParticles, Color color) {
  1912. for (int i = 0; i<playgroundParticles.particleCache.Length; i++) {
  1913. playgroundParticles.particleCache[i].color = color;
  1914. }
  1915. }
  1916. // Get vertices from a skinned world object in a Vector3-array
  1917. public static void GetPosition (SkinnedWorldObject particleStateWorldObject, bool updateNormals) {
  1918. if (updateNormals)
  1919. particleStateWorldObject.normals = particleStateWorldObject.mesh.normals;
  1920. Vector3[] vertices = particleStateWorldObject.mesh.vertices;
  1921. BoneWeight[] weights = particleStateWorldObject.mesh.boneWeights;
  1922. Matrix4x4[] bindPoses = particleStateWorldObject.mesh.bindposes;
  1923. Matrix4x4[] boneMatrices = new Matrix4x4[particleStateWorldObject.renderer.bones.Length];
  1924. int i = 0;
  1925. for (; i<boneMatrices.Length; i++) {
  1926. boneMatrices[i] = particleStateWorldObject.renderer.bones[i].localToWorldMatrix * bindPoses[i];
  1927. }
  1928. PlaygroundC.RunAsync(()=>{
  1929. Matrix4x4 vertexMatrix = new Matrix4x4();
  1930. for (i = 0; i<vertices.Length; i++) {
  1931. BoneWeight weight = weights[i];
  1932. Matrix4x4 m0 = boneMatrices[weight.boneIndex0];
  1933. Matrix4x4 m1 = boneMatrices[weight.boneIndex1];
  1934. Matrix4x4 m2 = boneMatrices[weight.boneIndex2];
  1935. Matrix4x4 m3 = boneMatrices[weight.boneIndex3];
  1936. for (int n=0;n<16;n++) {
  1937. vertexMatrix[n] =
  1938. m0[n] * weight.weight0 +
  1939. m1[n] * weight.weight1 +
  1940. m2[n] * weight.weight2 +
  1941. m3[n] * weight.weight3;
  1942. }
  1943. vertices[i] = vertexMatrix.MultiplyPoint3x4(vertices[i]);
  1944. }
  1945. particleStateWorldObject.vertexPositions = vertices;
  1946. });
  1947. }
  1948. // Get position from Mesh World Object
  1949. public static void GetPosition (Vector3[] v3, WorldObject particleStateWorldObject) {
  1950. if (particleStateWorldObject.meshFilter.sharedMesh!=particleStateWorldObject.mesh)
  1951. particleStateWorldObject.mesh = particleStateWorldObject.meshFilter.sharedMesh;
  1952. v3 = particleStateWorldObject.mesh.vertices;
  1953. }
  1954. // Get procedural position from Mesh World Object
  1955. public static void GetProceduralPosition (Vector3[] v3, WorldObject particleStateWorldObject) {
  1956. if (particleStateWorldObject.meshFilter.sharedMesh!=particleStateWorldObject.mesh)
  1957. particleStateWorldObject.mesh = particleStateWorldObject.meshFilter.sharedMesh;
  1958. Vector3[] vertices = particleStateWorldObject.mesh.vertices;
  1959. if (v3.Length!=vertices.Length) v3 = new Vector3[vertices.Length];
  1960. for (int i = 0; i<v3.Length; i++) {
  1961. v3[i] = particleStateWorldObject.transform.TransformPoint(vertices[i%vertices.Length]);
  1962. }
  1963. }
  1964. // Get normals from Mesh World Object
  1965. public static void GetNormals (Vector3[] v3, WorldObject particleStateWorldObject) {
  1966. v3 = particleStateWorldObject.mesh.normals;
  1967. }
  1968. // Set size for particles
  1969. public static void SetSize (PlaygroundParticlesC playgroundParticles, float size) {
  1970. for (int i = 0; i<playgroundParticles.particleCache.Length; i++) {
  1971. playgroundParticles.playgroundCache.initialSize[i] = size;
  1972. playgroundParticles.particleCache[i].size = size;
  1973. }
  1974. }
  1975. public void RefreshSystemRandom () {
  1976. internalRandom01 = new System.Random();
  1977. }
  1978. // Set random size for particles within sizeMinimum- and sizeMaximum range
  1979. public static void SetSizeRandom (PlaygroundParticlesC playgroundParticles, float sizeMinimum, float sizeMaximum) {
  1980. playgroundParticles.playgroundCache.initialSize = RandomFloat(playgroundParticles.particleCache.Length, sizeMinimum, sizeMaximum, playgroundParticles.internalRandom01);
  1981. playgroundParticles.sizeMin = sizeMinimum;
  1982. playgroundParticles.sizeMax = sizeMaximum;
  1983. playgroundParticles.previousSizeMin = playgroundParticles.sizeMin;
  1984. playgroundParticles.previousSizeMax = playgroundParticles.sizeMax;
  1985. }
  1986. // Set random rotation for particles within rotationMinimum- and rotationMaximum range
  1987. public static void SetRotationRandom (PlaygroundParticlesC playgroundParticles, float rotationMinimum, float rotationMaximum) {
  1988. playgroundParticles.playgroundCache.rotationSpeed = RandomFloat(playgroundParticles.particleCache.Length, rotationMinimum, rotationMaximum, playgroundParticles.internalRandom01);
  1989. for (int i = 0; i<playgroundParticles.particleCache.Length; i++) {
  1990. playgroundParticles.playgroundCache.rotation[i] = playgroundParticles.playgroundCache.initialRotation[i];
  1991. }
  1992. playgroundParticles.rotationSpeedMin = rotationMinimum;
  1993. playgroundParticles.rotationSpeedMax = rotationMaximum;
  1994. playgroundParticles.previousRotationSpeedMin = playgroundParticles.rotationSpeedMin;
  1995. playgroundParticles.previousRotationSpeedMax = playgroundParticles.rotationSpeedMax;
  1996. }
  1997. // Set random initial rotation for particles within rotationMinimum- and rotationMaximum range
  1998. public static void SetInitialRotationRandom (PlaygroundParticlesC playgroundParticles, float rotationMinimum, float rotationMaximum) {
  1999. playgroundParticles.playgroundCache.initialRotation = RandomFloat(playgroundParticles.particleCache.Length, rotationMinimum, rotationMaximum, playgroundParticles.internalRandom01);
  2000. for (int i = 0; i<playgroundParticles.particleCache.Length; i++) {
  2001. playgroundParticles.playgroundCache.rotation[i] = playgroundParticles.playgroundCache.initialRotation[i];
  2002. }
  2003. playgroundParticles.initialRotationMin = rotationMinimum;
  2004. playgroundParticles.initialRotationMax = rotationMaximum;
  2005. playgroundParticles.previousInitialRotationMin = playgroundParticles.initialRotationMin;
  2006. playgroundParticles.previousInitialRotationMax = playgroundParticles.initialRotationMax;
  2007. }
  2008. // Set initial random velocity for particles within velocityMinimum- and velocityMaximum range
  2009. public static void SetVelocityRandom (PlaygroundParticlesC playgroundParticles, Vector3 velocityMinimum, Vector3 velocityMaximum) {
  2010. playgroundParticles.playgroundCache.initialVelocity = new Vector3[playgroundParticles.particleCount];
  2011. for (int i = 0; i<playgroundParticles.particleCount; i++) {
  2012. if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.Spherical)
  2013. playgroundParticles.playgroundCache.initialVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, velocityMinimum.x, velocityMaximum.x);
  2014. else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.SphericalLinear)
  2015. playgroundParticles.playgroundCache.initialVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, velocityMinimum.x, velocityMaximum.x, (i*1f)/(playgroundParticles.particleCount*1f));
  2016. else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.RectangularLinear)
  2017. playgroundParticles.playgroundCache.initialVelocity[i] = Vector3.Lerp (velocityMinimum, velocityMaximum, (i*1f)/(playgroundParticles.particleCount*1f));
  2018. //else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.SphericalSector)
  2019. // playgroundParticles.playgroundCache.initialVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, playgroundParticles.initialVelocityMin.x, playgroundParticles.initialVelocityMax.x, playgroundParticles.initialVelocityMin.y, playgroundParticles.initialVelocityMax.y);
  2020. //else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.SphericalSectorLinear)
  2021. // playgroundParticles.playgroundCache.initialVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, playgroundParticles.initialVelocityMin.x, playgroundParticles.initialVelocityMax.x, playgroundParticles.initialVelocityMin.y, playgroundParticles.initialVelocityMax.y, (i*1f)/(playgroundParticles.particleCount*1f));
  2022. else playgroundParticles.playgroundCache.initialVelocity[i] = RandomRange(playgroundParticles.internalRandom01, velocityMinimum, velocityMaximum);
  2023. }
  2024. playgroundParticles.initialVelocityMin = velocityMinimum;
  2025. playgroundParticles.initialVelocityMax = velocityMaximum;
  2026. playgroundParticles.previousVelocityMin = playgroundParticles.initialVelocityMin;
  2027. playgroundParticles.previousVelocityMax = playgroundParticles.initialVelocityMax;
  2028. playgroundParticles.previousVelocityMethod = playgroundParticles.initialVelocityMethod;
  2029. }
  2030. // Set initial random local velocity for particles within velocityMinimum- and velocityMaximum range
  2031. public static void SetLocalVelocityRandom (PlaygroundParticlesC playgroundParticles, Vector3 velocityMinimum, Vector3 velocityMaximum) {
  2032. playgroundParticles.playgroundCache.initialLocalVelocity = new Vector3[playgroundParticles.particleCount];
  2033. for (int i = 0; i<playgroundParticles.particleCount; i++) {
  2034. if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.Spherical)
  2035. playgroundParticles.playgroundCache.initialLocalVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, velocityMinimum.x, velocityMaximum.x);
  2036. else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.SphericalLinear)
  2037. playgroundParticles.playgroundCache.initialLocalVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, velocityMinimum.x, velocityMaximum.x, (i*1f)/(playgroundParticles.particleCount*1f));
  2038. else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.RectangularLinear)
  2039. playgroundParticles.playgroundCache.initialLocalVelocity[i] = Vector3.Lerp (velocityMinimum, velocityMaximum, (i*1f)/(playgroundParticles.particleCount*1f));
  2040. //else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.SphericalSector)
  2041. // playgroundParticles.playgroundCache.initialLocalVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, playgroundParticles.initialLocalVelocityMin.x, playgroundParticles.initialLocalVelocityMax.x, playgroundParticles.initialLocalVelocityMin.y, playgroundParticles.initialLocalVelocityMax.y);
  2042. //else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.SphericalSectorLinear)
  2043. // playgroundParticles.playgroundCache.initialLocalVelocity[i] = RandomRangeSpherical(playgroundParticles.internalRandom01, playgroundParticles.initialLocalVelocityMin.x, playgroundParticles.initialLocalVelocityMax.x, playgroundParticles.initialLocalVelocityMin.y, playgroundParticles.initialLocalVelocityMax.y, (i*1f)/(playgroundParticles.particleCount*1f));
  2044. else playgroundParticles.playgroundCache.initialLocalVelocity[i] = RandomRange(playgroundParticles.internalRandom01, velocityMinimum, velocityMaximum);
  2045. }
  2046. playgroundParticles.initialLocalVelocityMin = velocityMinimum;
  2047. playgroundParticles.initialLocalVelocityMax = velocityMaximum;
  2048. playgroundParticles.previousLocalVelocityMin = playgroundParticles.initialLocalVelocityMin;
  2049. playgroundParticles.previousLocalVelocityMax = playgroundParticles.initialLocalVelocityMax;
  2050. playgroundParticles.previousLocalVelocityMethod = playgroundParticles.initialLocalVelocityMethod;
  2051. }
  2052. // Set material for particle system
  2053. public static void SetMaterial (PlaygroundParticlesC playgroundParticles, Material particleMaterial) {
  2054. playgroundParticles.particleSystemRenderer.sharedMaterial = particleMaterial;
  2055. }
  2056. // Set alphas for particles
  2057. public static void SetAlpha (PlaygroundParticlesC playgroundParticles, float alpha) {
  2058. Color pColor;
  2059. for (int i = 0; i<playgroundParticles.particleCache.Length; i++) {
  2060. pColor = playgroundParticles.particleCache[i].color;
  2061. pColor.a = alpha;
  2062. playgroundParticles.particleCache[i].color = pColor;
  2063. }
  2064. }
  2065. // Move all particles in direction
  2066. public static void Translate (PlaygroundParticlesC playgroundParticles, Vector3 direction) {
  2067. for (int i = 0; i<playgroundParticles.particleCache.Length; i++)
  2068. playgroundParticles.particleCache[i].position += direction;
  2069. }
  2070. // Add new state from state
  2071. public static void Add (PlaygroundParticlesC playgroundParticles, ParticleStateC state) {
  2072. playgroundParticles.states.Add(state);
  2073. state.Initialize();
  2074. }
  2075. // Add new state from image
  2076. public static void Add (PlaygroundParticlesC playgroundParticles, Texture2D image, float scale, Vector3 offset, string stateName, Transform stateTransform) {
  2077. playgroundParticles.states.Add(new ParticleStateC());
  2078. playgroundParticles.states[playgroundParticles.states.Count-1].ConstructParticles(image,scale,offset,stateName,stateTransform);
  2079. }
  2080. // Add new state from image with depthmap
  2081. public static void Add (PlaygroundParticlesC playgroundParticles, Texture2D image, Texture2D depthmap, float depthmapStrength, float scale, Vector3 offset, string stateName, Transform stateTransform) {
  2082. playgroundParticles.states.Add(new ParticleStateC());
  2083. playgroundParticles.states[playgroundParticles.states.Count-1].ConstructParticles(image,scale,offset,stateName,stateTransform);
  2084. playgroundParticles.states[playgroundParticles.states.Count-1].stateDepthmap = depthmap;
  2085. playgroundParticles.states[playgroundParticles.states.Count-1].stateDepthmapStrength = depthmapStrength;
  2086. }
  2087. // Destroy a PlaygroundParticlesC object
  2088. public static void Destroy (PlaygroundParticlesC playgroundParticles) {
  2089. Clear(playgroundParticles);
  2090. MonoBehaviour.DestroyImmediate(playgroundParticles.particleSystemGameObject);
  2091. playgroundParticles = null;
  2092. }
  2093. // Sorts the particles in lifetime
  2094. bool isSettingLifetime = false;
  2095. public static void SetLifetime (PlaygroundParticlesC playgroundParticles, SORTINGC sorting, float time) {
  2096. if (!playgroundParticles.enabled) return;
  2097. if (playgroundParticles.isSettingLifetime || playgroundParticles.isSettingParticleCount) return;
  2098. if (playgroundParticles.internalRandom01==null)
  2099. playgroundParticles.RefreshSystemRandom();
  2100. SetLifetimeThreadSafe (playgroundParticles, sorting, time);
  2101. }
  2102. public static void SetLifetimeThreadSafe (PlaygroundParticlesC playgroundParticles, SORTINGC sorting, float time) {
  2103. playgroundParticles.isSettingLifetime = true;
  2104. PlaygroundC.RunAsync(()=>{
  2105. playgroundParticles.lifetime = time;
  2106. SetLifetimeSubtraction(playgroundParticles);
  2107. playgroundParticles.playgroundCache.lifetimeOffset = new float[playgroundParticles.particleCount];
  2108. int pCount = playgroundParticles.playgroundCache.lifetimeOffset.Length;
  2109. if (playgroundParticles.source!=SOURCEC.Script) {
  2110. float lifetimeEmission = playgroundParticles.lifetimeEmission;
  2111. if (!playgroundParticles.loop)
  2112. lifetimeEmission*=.95f;
  2113. switch (sorting) {
  2114. case SORTINGC.Scrambled:
  2115. for (int r = 0; r<playgroundParticles.particleCount; r++) {
  2116. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2117. playgroundParticles.playgroundCache.lifetimeOffset[r] = RandomRange(playgroundParticles.internalRandom01, 0f, playgroundParticles.lifetime*lifetimeEmission);
  2118. }
  2119. break;
  2120. case SORTINGC.ScrambledLinear:
  2121. float slPerc;
  2122. for (int sl = 0; sl<playgroundParticles.particleCount; sl++) {
  2123. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2124. slPerc = (sl*1f)/(playgroundParticles.particleCount*1f);
  2125. playgroundParticles.playgroundCache.lifetimeOffset[sl] = playgroundParticles.lifetime*lifetimeEmission*slPerc;
  2126. }
  2127. for (int i = playgroundParticles.playgroundCache.lifetimeOffset.Length-1; i>0; i--) {
  2128. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2129. int r = playgroundParticles.internalRandom01.Next(0,i);
  2130. float tmp = playgroundParticles.playgroundCache.lifetimeOffset[i];
  2131. playgroundParticles.playgroundCache.lifetimeOffset[i] = playgroundParticles.playgroundCache.lifetimeOffset[r];
  2132. playgroundParticles.playgroundCache.lifetimeOffset[r] = tmp;
  2133. }
  2134. break;
  2135. case SORTINGC.Burst:
  2136. // No action needed for spawning all particles at once
  2137. break;
  2138. case SORTINGC.Linear:
  2139. float lPerc;
  2140. for (int l = 0; l<playgroundParticles.particleCount; l++) {
  2141. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2142. lPerc = (l*1f)/(playgroundParticles.particleCount*1f);
  2143. playgroundParticles.playgroundCache.lifetimeOffset[l] = playgroundParticles.lifetime*lifetimeEmission*lPerc;
  2144. }
  2145. break;
  2146. case SORTINGC.Reversed:
  2147. float rPerc;
  2148. int rInc = 0;
  2149. for (int r = playgroundParticles.particleCount-1; r>=0; r--) {
  2150. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2151. rPerc = (rInc*1f)/(playgroundParticles.particleCount*1f);
  2152. rInc++;
  2153. playgroundParticles.playgroundCache.lifetimeOffset[r] = playgroundParticles.lifetime*lifetimeEmission*rPerc;
  2154. }
  2155. break;
  2156. case SORTINGC.NearestNeighborReversed:
  2157. playgroundParticles.nearestNeighborOrigin = Mathf.Clamp(playgroundParticles.nearestNeighborOrigin, 0, playgroundParticles.particleCount-1);
  2158. float[] nnDist = new float[playgroundParticles.particleCount];
  2159. float nnHighest = 0;
  2160. Vector3 nnrOrigin = Vector3.zero;
  2161. switch (playgroundParticles.nearestNeighborOriginMethod) {
  2162. case NEARESTNEIGHBORORIGINMETHOD.SourcePoint: nnrOrigin = playgroundParticles.playgroundCache.targetPosition[playgroundParticles.nearestNeighborOrigin%playgroundParticles.particleCount]; break;
  2163. case NEARESTNEIGHBORORIGINMETHOD.Vector3: nnrOrigin = playgroundParticles.nearestNeighborOriginVector3; break;
  2164. case NEARESTNEIGHBORORIGINMETHOD.Transform: nnrOrigin = playgroundParticles.nearestNeighborOriginTransformPosition; break;
  2165. }
  2166. for (int nn = 0; nn<playgroundParticles.particleCount; nn++) {
  2167. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2168. nnDist[nn%playgroundParticles.particleCount] = Vector3.SqrMagnitude(nnrOrigin - playgroundParticles.playgroundCache.targetPosition[nn%playgroundParticles.particleCount]);
  2169. if (nnDist[nn%playgroundParticles.particleCount]>nnHighest)
  2170. nnHighest = nnDist[nn%playgroundParticles.particleCount];
  2171. }
  2172. if (nnHighest>0) {
  2173. for (int nn = 0; nn<playgroundParticles.particleCount; nn++) {
  2174. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2175. playgroundParticles.playgroundCache.lifetimeOffset[nn%playgroundParticles.particleCount] = Mathf.Lerp(playgroundParticles.lifetime*(lifetimeEmission-.001f), 0, (nnDist[nn%playgroundParticles.particleCount]/nnHighest));
  2176. }
  2177. } else {
  2178. for (int nn = 0; nn<playgroundParticles.particleCount; nn++) {
  2179. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2180. playgroundParticles.playgroundCache.lifetimeOffset[nn%playgroundParticles.particleCount] = 0;
  2181. }
  2182. }
  2183. break;
  2184. case SORTINGC.NearestNeighbor:
  2185. playgroundParticles.nearestNeighborOrigin = Mathf.Clamp(playgroundParticles.nearestNeighborOrigin, 0, playgroundParticles.particleCount-1);
  2186. float[] nnrDist = new float[playgroundParticles.particleCount];
  2187. float nnrHighest = 0;
  2188. Vector3 nnOrigin = Vector3.zero;
  2189. switch (playgroundParticles.nearestNeighborOriginMethod) {
  2190. case NEARESTNEIGHBORORIGINMETHOD.SourcePoint: nnOrigin = playgroundParticles.playgroundCache.targetPosition[playgroundParticles.nearestNeighborOrigin%playgroundParticles.particleCount]; break;
  2191. case NEARESTNEIGHBORORIGINMETHOD.Vector3: nnOrigin = playgroundParticles.nearestNeighborOriginVector3; break;
  2192. case NEARESTNEIGHBORORIGINMETHOD.Transform: nnOrigin = playgroundParticles.nearestNeighborOriginTransformPosition; break;
  2193. }
  2194. for (int nnr = 0; nnr<playgroundParticles.particleCount; nnr++) {
  2195. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2196. nnrDist[nnr%playgroundParticles.particleCount] = Vector3.SqrMagnitude(nnOrigin-playgroundParticles.playgroundCache.targetPosition[nnr%playgroundParticles.particleCount]);
  2197. if (nnrDist[nnr%playgroundParticles.particleCount]>nnrHighest)
  2198. nnrHighest = nnrDist[nnr%playgroundParticles.particleCount];
  2199. }
  2200. if (nnrHighest>0) {
  2201. for (int nnr = 0; nnr<playgroundParticles.particleCount; nnr++) {
  2202. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2203. playgroundParticles.playgroundCache.lifetimeOffset[nnr%playgroundParticles.particleCount] = Mathf.Lerp(0, playgroundParticles.lifetime*(lifetimeEmission-.001f), (nnrDist[nnr%playgroundParticles.particleCount]/nnrHighest));
  2204. }
  2205. } else {
  2206. for (int nnr = 0; nnr<playgroundParticles.particleCount; nnr++) {
  2207. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2208. playgroundParticles.playgroundCache.lifetimeOffset[nnr%playgroundParticles.particleCount] = 0;
  2209. }
  2210. }
  2211. break;
  2212. case SORTINGC.Custom:
  2213. for (int cs = playgroundParticles.particleCount-1; cs>=0; cs--) {
  2214. if (pCount!=playgroundParticles.playgroundCache.lifetimeOffset.Length) {playgroundParticles.isSettingLifetime = false; return;}
  2215. playgroundParticles.playgroundCache.lifetimeOffset[cs] = playgroundParticles.lifetime*playgroundParticles.lifetimeEmission*playgroundParticles.lifetimeSorting.Evaluate(cs*1f/playgroundParticles.particleCount*1f);
  2216. }
  2217. break;
  2218. }
  2219. }
  2220. SetEmissionRate(playgroundParticles);
  2221. SetParticleTimeNow(playgroundParticles);
  2222. playgroundParticles.previousLifetime = playgroundParticles.lifetime;
  2223. playgroundParticles.previousLifetimeEmission = playgroundParticles.lifetimeEmission;
  2224. playgroundParticles.previousNearestNeighborOrigin = playgroundParticles.nearestNeighborOrigin;
  2225. playgroundParticles.previousNearestNeighborOriginMethod = playgroundParticles.nearestNeighborOriginMethod;
  2226. playgroundParticles.previousNearestNeighborOriginVector3 = playgroundParticles.nearestNeighborOriginVector3;
  2227. playgroundParticles.previousNearestNeighborOriginTransform = playgroundParticles.nearestNeighborOriginTransform;
  2228. playgroundParticles.previousSorting = playgroundParticles.sorting;
  2229. playgroundParticles.isDoneThread = true;
  2230. playgroundParticles.isSettingLifetime = false;
  2231. playgroundParticles.localDeltaTime = 0f;
  2232. playgroundParticles.localTime = PlaygroundC.globalTime;
  2233. playgroundParticles.lastTimeUpdated = PlaygroundC.globalTime;
  2234. });
  2235. }
  2236. public static void SetLifetimeSubtraction (PlaygroundParticlesC playgroundParticles) {
  2237. playgroundParticles.playgroundCache.lifetimeSubtraction = new float[playgroundParticles.particleCount];
  2238. System.Random random = new System.Random();
  2239. for (int p = 0; p<playgroundParticles.playgroundCache.lifetimeSubtraction.Length; p++) {
  2240. if (playgroundParticles.lifetimeValueMethod==VALUEMETHOD.Constant) {
  2241. playgroundParticles.playgroundCache.lifetimeSubtraction[p] = 0;
  2242. } else {
  2243. playgroundParticles.playgroundCache.lifetimeSubtraction[p] = RandomRange(random, 0, playgroundParticles.lifetime-playgroundParticles.lifetimeMin);
  2244. }
  2245. }
  2246. playgroundParticles.previousLifetimeMin = playgroundParticles.lifetimeMin;
  2247. playgroundParticles.previousLifetimeValueMethod = playgroundParticles.lifetimeValueMethod;
  2248. }
  2249. // Set emission rate percentage of particle count
  2250. public static void SetEmissionRate (PlaygroundParticlesC playgroundParticles) {
  2251. if (playgroundParticles.playgroundCache.emission==null) return;
  2252. float rateCount = playgroundParticles.lifetime*playgroundParticles.emissionRate;
  2253. int currentCount = playgroundParticles.playgroundCache.emission.Length;
  2254. bool hasOverflow = playgroundParticles.source!=SOURCEC.Transform&&(playgroundParticles.overflowOffset!=Vector3.zero||playgroundParticles.applySourceScatter&&(playgroundParticles.sourceScatterMin!=Vector3.zero||playgroundParticles.sourceScatterMax!=Vector3.zero));
  2255. for (int p = 0; p<playgroundParticles.playgroundCache.emission.Length; p++) {
  2256. if (currentCount!=playgroundParticles.playgroundCache.emission.Length || playgroundParticles.playgroundCache.lifetimeOffset.Length!=currentCount) return;
  2257. if (playgroundParticles.emissionRate!=0 && playgroundParticles.source!=SOURCEC.Script) {
  2258. if (playgroundParticles.sorting!=SORTINGC.Burst || playgroundParticles.sorting==SORTINGC.NearestNeighbor && hasOverflow || playgroundParticles.sorting==SORTINGC.NearestNeighborReversed && hasOverflow) {
  2259. playgroundParticles.playgroundCache.emission[p] = (playgroundParticles.playgroundCache.lifetimeOffset[p]>=playgroundParticles.lifetime-rateCount && playgroundParticles.emit);
  2260. } else {
  2261. playgroundParticles.playgroundCache.emission[p] = (playgroundParticles.emit && playgroundParticles.emissionRate>((p*1f)/(currentCount*1f)));
  2262. }
  2263. } else playgroundParticles.playgroundCache.emission[p] = false;
  2264. if (playgroundParticles.playgroundCache.emission[p]) {
  2265. playgroundParticles.playgroundCache.rebirth[p] = true;
  2266. playgroundParticles.playgroundCache.simulate[p] = true;
  2267. } else if (playgroundParticles.source==SOURCEC.Script)
  2268. playgroundParticles.playgroundCache.rebirth[p] = false;
  2269. }
  2270. playgroundParticles.previousEmissionRate = playgroundParticles.emissionRate;
  2271. playgroundParticles.hasActiveParticles = true;
  2272. playgroundParticles.threadHadNoActiveParticles = false;
  2273. }
  2274. // Set time of particles with current time
  2275. public static void SetParticleTimeNow (PlaygroundParticlesC playgroundParticles) {
  2276. if (playgroundParticles.playgroundCache.lifetimeOffset==null || playgroundParticles.playgroundCache.lifetimeOffset.Length!=playgroundParticles.particleCount) return;
  2277. if (playgroundParticles.playgroundCache.life==null || playgroundParticles.playgroundCache.life.Length!=playgroundParticles.particleCount) return;
  2278. playgroundParticles.isSettingParticleTime = true;
  2279. if (playgroundParticles.source!=SOURCEC.Script) {
  2280. float currentTime = PlaygroundC.globalTime+playgroundParticles.lifetimeOffset;
  2281. if (currentTime<=0) currentTime = playgroundParticles.localTime+.1f;
  2282. for (int p = 0; p<playgroundParticles.particleCount; p++) {
  2283. playgroundParticles.playgroundCache.birth[p] = (currentTime+playgroundParticles.playgroundCache.lifetimeOffset[p])-playgroundParticles.lifetime;
  2284. playgroundParticles.playgroundCache.death[p] = playgroundParticles.playgroundCache.birth[p]+playgroundParticles.lifetime;
  2285. playgroundParticles.playgroundCache.life[p] = (playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.lifetimeOffset[p])-playgroundParticles.playgroundCache.birth[p];
  2286. playgroundParticles.particleCache[p].startLifetime = playgroundParticles.lifetime;
  2287. playgroundParticles.particleCache[p].size = 0;
  2288. playgroundParticles.playgroundCache.targetPosition[p] = PlaygroundC.initialTargetPosition;
  2289. playgroundParticles.playgroundCache.previousTargetPosition[p] = PlaygroundC.initialTargetPosition;
  2290. playgroundParticles.playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  2291. playgroundParticles.playgroundCache.simulate[p] = true;
  2292. playgroundParticles.playgroundCache.emission[p] = true;
  2293. playgroundParticles.playgroundCache.isFirstLoop[p] = true;
  2294. playgroundParticles.playgroundCache.isNonBirthed[p] = true;
  2295. playgroundParticles.playgroundCache.isCalculatedThisFrame[p] = false;
  2296. if (!playgroundParticles.isPrewarming)
  2297. playgroundParticles.playgroundCache.rebirth[p] = true;
  2298. }
  2299. playgroundParticles.localTime = currentTime;
  2300. playgroundParticles.lastTimeUpdated = playgroundParticles.localTime;
  2301. playgroundParticles.localDeltaTime = 0;
  2302. playgroundParticles.t = 0;
  2303. playgroundParticles.simulationStarted = playgroundParticles.localTime;
  2304. playgroundParticles.cameFromNonEmissionFrame = true;
  2305. playgroundParticles.hasActiveParticles = true;
  2306. playgroundParticles.threadHadNoActiveParticles = false;
  2307. }
  2308. playgroundParticles.isSettingParticleTime = false;
  2309. }
  2310. bool isSettingParticleTime = false;
  2311. // Set time of particles with current time and take emission delta into consideration (time when emit went false to current)
  2312. public static void SetParticleTimeNowWithRestEmission (PlaygroundParticlesC playgroundParticles) {
  2313. playgroundParticles.isSettingParticleTime = true;
  2314. float currentTime = playgroundParticles.localTime;
  2315. float emissionDelta = currentTime-playgroundParticles.emissionStopped;
  2316. if (playgroundParticles.loop && emissionDelta<playgroundParticles.lifetime && emissionDelta>0) {
  2317. playgroundParticles.cameFromNonEmissionFrame = true;
  2318. } else {
  2319. SetParticleTimeNow(playgroundParticles);
  2320. return;
  2321. }
  2322. for (int p = 0; p<playgroundParticles.particleCount; p++) {
  2323. playgroundParticles.playgroundCache.birthDelay[p] = emissionDelta;
  2324. playgroundParticles.playgroundCache.rebirth[p] = true;
  2325. playgroundParticles.playgroundCache.simulate[p] = true;
  2326. playgroundParticles.playgroundCache.emission[p] = true;
  2327. }
  2328. playgroundParticles.isSettingParticleTime = false;
  2329. }
  2330. // Get color from evaluated lifetime color value where time is normalized
  2331. public static Color32 GetColorAtLifetime (PlaygroundParticlesC playgroundParticles, float time) {
  2332. return playgroundParticles.lifetimeColor.Evaluate(time/playgroundParticles.lifetime);
  2333. }
  2334. // Color all particles from evaluated lifetime color value where time is normalized
  2335. public static void SetColorAtLifetime (PlaygroundParticlesC playgroundParticles, float time) {
  2336. Color32 c = playgroundParticles.lifetimeColor.Evaluate(time/playgroundParticles.lifetime);
  2337. for (int p = 0; p<playgroundParticles.particleCount; p++)
  2338. playgroundParticles.particleCache[p].color = c;
  2339. playgroundParticles.shurikenParticleSystem.SetParticles(playgroundParticles.particleCache, playgroundParticles.particleCache.Length);
  2340. }
  2341. // Color all particles from lifetime span with sorting
  2342. public static void SetColorWithLifetimeSorting (PlaygroundParticlesC playgroundParticles) {
  2343. SetLifetime(playgroundParticles, playgroundParticles.sorting, playgroundParticles.lifetime);
  2344. Color32 c;
  2345. for (int p = 0; p<playgroundParticles.particleCount; p++) {
  2346. c = playgroundParticles.lifetimeColor.Evaluate(playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime);
  2347. playgroundParticles.particleCache[p].color = c;
  2348. }
  2349. playgroundParticles.shurikenParticleSystem.SetParticles(playgroundParticles.particleCache, playgroundParticles.particleCache.Length);
  2350. }
  2351. public bool IsSettingParticleCount () {
  2352. return isSettingParticleCount;
  2353. }
  2354. // Sets the amount of particles and initiates the necessary arrays
  2355. bool isSettingParticleCount = false;
  2356. public static void SetParticleCount (PlaygroundParticlesC playgroundParticles, int amount) {
  2357. if (playgroundParticles.isSettingParticleCount) return;
  2358. if (playgroundParticles.isPrewarming) return;
  2359. if (amount<0) amount = 0;
  2360. if (playgroundParticles.internalRandom01==null)
  2361. playgroundParticles.RefreshSystemRandom();
  2362. playgroundParticles.particleCount = amount;
  2363. playgroundParticles.previousParticleCount = amount;
  2364. // Create Particles
  2365. playgroundParticles.particleCache = new ParticleSystem.Particle[amount];
  2366. playgroundParticles.shurikenParticleSystem.Emit(amount);
  2367. playgroundParticles.shurikenParticleSystem.GetParticles(playgroundParticles.particleCache);
  2368. // Clear collision cache
  2369. playgroundParticles.collisionCache = null;
  2370. playgroundParticles.inTransition = false;
  2371. playgroundParticles.hasActiveParticles = true;
  2372. playgroundParticles.threadHadNoActiveParticles = false;
  2373. playgroundParticles.isSettingParticleCount = true;
  2374. PlaygroundC.RunAsync(()=>{
  2375. if (!playgroundParticles.isSettingParticleCount) return;
  2376. if (playgroundParticles.playgroundCache==null)
  2377. playgroundParticles.playgroundCache = new PlaygroundCache();
  2378. // Rebuild Cache
  2379. playgroundParticles.playgroundCache.size = new float[amount];
  2380. playgroundParticles.playgroundCache.birth = new float[amount];
  2381. playgroundParticles.playgroundCache.death = new float[amount];
  2382. playgroundParticles.playgroundCache.rebirth = new bool[amount];
  2383. playgroundParticles.playgroundCache.birthDelay = new float[amount];
  2384. playgroundParticles.playgroundCache.life = new float[amount];
  2385. playgroundParticles.playgroundCache.lifetimeSubtraction = new float[amount];
  2386. playgroundParticles.playgroundCache.rotation = new float[amount];
  2387. playgroundParticles.playgroundCache.lifetimeOffset = new float[amount];
  2388. playgroundParticles.playgroundCache.emission = new bool[amount];
  2389. playgroundParticles.playgroundCache.changedByProperty = new bool[amount];
  2390. playgroundParticles.playgroundCache.changedByPropertyColor = new bool[amount];
  2391. playgroundParticles.playgroundCache.changedByPropertyColorLerp = new bool[amount];
  2392. playgroundParticles.playgroundCache.changedByPropertyColorKeepAlpha = new bool[amount];
  2393. playgroundParticles.playgroundCache.changedByPropertySize = new bool[amount];
  2394. playgroundParticles.playgroundCache.changedByPropertyTarget = new bool[amount];
  2395. playgroundParticles.playgroundCache.changedByPropertyDeath = new bool[amount];
  2396. playgroundParticles.playgroundCache.propertyTarget = new int[amount];
  2397. playgroundParticles.playgroundCache.propertyId = new int[amount];
  2398. playgroundParticles.playgroundCache.excludeFromManipulatorId = new int[amount];
  2399. playgroundParticles.playgroundCache.propertyColorId = new int[amount];
  2400. playgroundParticles.playgroundCache.manipulatorId = new int[amount];
  2401. playgroundParticles.playgroundCache.color = new Color32[amount];
  2402. playgroundParticles.playgroundCache.scriptedColor = new Color32[amount];
  2403. playgroundParticles.playgroundCache.initialColor = new Color32[amount];
  2404. playgroundParticles.playgroundCache.lifetimeColorId = new int[amount];
  2405. playgroundParticles.playgroundCache.noForce = new bool[amount];
  2406. playgroundParticles.playgroundCache.position = new Vector3[amount];
  2407. playgroundParticles.playgroundCache.targetPosition = new Vector3[amount];
  2408. playgroundParticles.playgroundCache.targetDirection = new Vector3[amount];
  2409. playgroundParticles.playgroundCache.previousTargetPosition = new Vector3[amount];
  2410. playgroundParticles.playgroundCache.previousParticlePosition = new Vector3[amount];
  2411. playgroundParticles.playgroundCache.collisionParticlePosition = new Vector3[amount];
  2412. playgroundParticles.playgroundCache.localSpaceMovementCompensation = new Vector3[amount];
  2413. playgroundParticles.playgroundCache.scatterPosition = new Vector3[amount];
  2414. playgroundParticles.playgroundCache.velocity = new Vector3[amount];
  2415. playgroundParticles.playgroundCache.isMasked = new bool[amount];
  2416. playgroundParticles.playgroundCache.maskAlpha = new float[amount];
  2417. playgroundParticles.playgroundCache.isNonBirthed = new bool[amount];
  2418. playgroundParticles.playgroundCache.isFirstLoop = new bool[amount];
  2419. playgroundParticles.playgroundCache.simulate = new bool[amount];
  2420. playgroundParticles.playgroundCache.isCalculatedThisFrame = new bool[amount];
  2421. playgroundParticles.playgroundCache.maskSorting = null;
  2422. for (int i = 0; i<amount; i++) {
  2423. playgroundParticles.particleCache[i].size = 0f;
  2424. playgroundParticles.playgroundCache.rebirth[i] = true;
  2425. playgroundParticles.playgroundCache.simulate[i] = true;
  2426. playgroundParticles.playgroundCache.isNonBirthed[i] = true;
  2427. playgroundParticles.playgroundCache.isFirstLoop[i] = true;
  2428. // Set color gradient id
  2429. if (playgroundParticles.colorSource==COLORSOURCEC.LifetimeColors && playgroundParticles.lifetimeColors.Count>0) {
  2430. playgroundParticles.lifetimeColorId++;playgroundParticles.lifetimeColorId=playgroundParticles.lifetimeColorId%playgroundParticles.lifetimeColors.Count;
  2431. playgroundParticles.playgroundCache.lifetimeColorId[i] = playgroundParticles.lifetimeColorId;
  2432. }
  2433. }
  2434. // Set sizes
  2435. SetSizeRandom(playgroundParticles, playgroundParticles.sizeMin, playgroundParticles.sizeMax);
  2436. playgroundParticles.previousSizeMin = playgroundParticles.sizeMin;
  2437. playgroundParticles.previousSizeMax = playgroundParticles.sizeMax;
  2438. // Set rotations
  2439. playgroundParticles.playgroundCache.initialRotation = RandomFloat(amount, playgroundParticles.initialRotationMin, playgroundParticles.initialRotationMax, playgroundParticles.internalRandom01);
  2440. playgroundParticles.playgroundCache.rotationSpeed = RandomFloat(amount, playgroundParticles.rotationSpeedMin, playgroundParticles.rotationSpeedMax, playgroundParticles.internalRandom01);
  2441. playgroundParticles.previousInitialRotationMin = playgroundParticles.initialRotationMin;
  2442. playgroundParticles.previousInitialRotationMax = playgroundParticles.initialRotationMax;
  2443. playgroundParticles.previousRotationSpeedMin = playgroundParticles.rotationSpeedMin;
  2444. playgroundParticles.previousRotationSpeedMax = playgroundParticles.rotationSpeedMax;
  2445. // Set velocities
  2446. SetVelocityRandom(playgroundParticles, playgroundParticles.initialVelocityMin, playgroundParticles.initialVelocityMax);
  2447. SetLocalVelocityRandom(playgroundParticles, playgroundParticles.initialLocalVelocityMin, playgroundParticles.initialLocalVelocityMax);
  2448. // Set Emission
  2449. Emission(playgroundParticles, playgroundParticles.emit, false);
  2450. // Make sure scatter is available first lifetime cycle
  2451. if (playgroundParticles.applySourceScatter)
  2452. playgroundParticles.RefreshScatter();
  2453. playgroundParticles.isDoneThread = true;
  2454. playgroundParticles.isSettingParticleCount = false;
  2455. });
  2456. }
  2457. bool prevLoop;
  2458. SOURCEC prevSource;
  2459. /// <summary>
  2460. /// Updates a PlaygroundParticlesC object (called each calculation step from PlaygroundC).
  2461. /// </summary>
  2462. public bool UpdateSystem () {
  2463. if (isYieldRefreshing || isLoading || playgroundCache==null || thisInstance==null || !particleSystemGameObject.activeInHierarchy || !particleSystemGameObject.activeSelf || !enabled || isSettingLifetime || isSettingParticleCount || isPrewarming) return false;
  2464. // Emission halt for disabling called from calculation thread
  2465. if (queueEmissionHalt && !IsAlive() && !loop && loopExceeded && hasEmitted) {
  2466. if (disableOnDoneRoutine==ONDONE.Inactivate) {
  2467. particleSystemGameObject.SetActive(false);
  2468. return false;
  2469. } else if (Application.isPlaying) {
  2470. DestroyImmediate(particleSystemGameObject);
  2471. return false;
  2472. }
  2473. }
  2474. // Check that particle system has emitted (used for Disable On Done routines)
  2475. if (emit)
  2476. hasEmitted = true;
  2477. // Source
  2478. if (source!=prevSource) {
  2479. prevSource=source;
  2480. return false;
  2481. }
  2482. // Particle count
  2483. if (particleCount!=previousParticleCount) {
  2484. SetParticleCount(thisInstance, particleCount);
  2485. isDoneThread = true;
  2486. if (particleCount>0)
  2487. StartCoroutine(Boot());
  2488. return false;
  2489. }
  2490. // Particle emission
  2491. if (emit!=previousEmission) {
  2492. Emit(emit);
  2493. return false;
  2494. }
  2495. // Particle loop enabled again
  2496. if (loop!=previousLoop) {
  2497. if (emit && loop)
  2498. Emit(true);
  2499. previousLoop = loop;
  2500. return false;
  2501. }
  2502. // Particle size
  2503. if (sizeMin!=previousSizeMin || sizeMax!=previousSizeMax)
  2504. SetSizeRandom(thisInstance, sizeMin, sizeMax);
  2505. // Particle rotation
  2506. if (initialRotationMin!=previousInitialRotationMin || initialRotationMax!=previousInitialRotationMax)
  2507. SetInitialRotationRandom(thisInstance, initialRotationMin, initialRotationMax);
  2508. if (rotationSpeedMin!=previousRotationSpeedMin || rotationSpeedMax!=previousRotationSpeedMax)
  2509. SetRotationRandom(thisInstance, rotationSpeedMin, rotationSpeedMax);
  2510. // Particle velocity
  2511. if (applyInitialVelocity) {
  2512. if (initialVelocityMin!=previousVelocityMin || initialVelocityMax!=previousVelocityMax || initialVelocityMethod!=previousVelocityMethod || playgroundCache.initialVelocity==null || playgroundCache.initialVelocity.Length!=particleCount) {
  2513. SetVelocityRandom(thisInstance, initialVelocityMin, initialVelocityMax);
  2514. return false;
  2515. }
  2516. }
  2517. // Particle local velocity
  2518. if (applyInitialLocalVelocity) {
  2519. if (initialLocalVelocityMin!=previousLocalVelocityMin || initialLocalVelocityMax!=previousLocalVelocityMax || initialLocalVelocityMethod!=previousLocalVelocityMethod || playgroundCache.initialLocalVelocity==null || playgroundCache.initialLocalVelocity.Length!=particleCount) {
  2520. SetLocalVelocityRandom(thisInstance, initialLocalVelocityMin, initialLocalVelocityMax);
  2521. return false;
  2522. }
  2523. }
  2524. // Particle life
  2525. if (previousLifetime!=lifetime ||
  2526. lifetimeValueMethod==VALUEMETHOD.Constant && previousLifetimeValueMethod!=lifetimeValueMethod) {
  2527. Start();
  2528. return false;
  2529. }
  2530. // Lifetime sorting
  2531. if (previousNearestNeighborOriginMethod!=nearestNeighborOriginMethod ||
  2532. (sorting==SORTINGC.NearestNeighbor||sorting==SORTINGC.NearestNeighborReversed)&&
  2533. (nearestNeighborOriginMethod==NEARESTNEIGHBORORIGINMETHOD.SourcePoint && previousNearestNeighborOrigin!=nearestNeighborOrigin ||
  2534. nearestNeighborOriginMethod==NEARESTNEIGHBORORIGINMETHOD.Vector3 && previousNearestNeighborOriginVector3!=nearestNeighborOriginVector3 ||
  2535. nearestNeighborOriginMethod==NEARESTNEIGHBORORIGINMETHOD.Transform && previousNearestNeighborOriginTransform!=nearestNeighborOriginTransform) ||
  2536. previousSorting!=sorting || previousLifetimeEmission!=lifetimeEmission) {
  2537. Start();
  2538. return false;
  2539. }
  2540. // Particle total lifetime
  2541. if (lifetimeValueMethod==VALUEMETHOD.RandomBetweenTwoValues) {
  2542. if (previousLifetimeMin!=lifetimeMin ||
  2543. previousLifetime!=lifetime ||
  2544. previousLifetimeValueMethod!=lifetimeValueMethod) {
  2545. SetLifetimeSubtraction(thisInstance);
  2546. previousLifetimeMin=lifetimeMin;
  2547. }
  2548. }
  2549. // Particle emission rate
  2550. if (previousEmissionRate!=emissionRate && source!=SOURCEC.Script) {
  2551. SetEmissionRate(thisInstance);
  2552. }
  2553. // Particle state change
  2554. if (source==SOURCEC.State && activeState!=previousActiveState) {
  2555. if (states[activeState].positionLength>particleCount)
  2556. SetParticleCount(thisInstance, states[activeState].positionLength);
  2557. previousActiveState = activeState;
  2558. }
  2559. // All good!
  2560. return true;
  2561. }
  2562. /// <summary>
  2563. /// Updates the Shuriken component. This sets all particles previously calculated and syncs particle positioning if syncPositionsOnMainThread is true.
  2564. /// </summary>
  2565. public void UpdateShuriken () {
  2566. // Early out if no update should occur
  2567. if (!isReadyForThreadedCalculations || !calculate) return;
  2568. // Collision detection (runs on main-thread)
  2569. if (collision)
  2570. Collisions(thisInstance);
  2571. // Sync positions to main-thread
  2572. if (syncPositionsOnMainThread || collision || (forceSkinnedMeshUpdateOnMainThread && source==SOURCEC.SkinnedWorldObject && (onlySourcePositioning||onlyLifetimePositioning))) {
  2573. if (particleCache.Length!=playgroundCache.position.Length)
  2574. return;
  2575. if (playgroundCache.life.Length == particleCount && playgroundCache.simulate.Length == particleCount) {
  2576. bool setSource = onlySourcePositioning && !cancelDeltaPositioningOnSync;
  2577. float mslt = minShurikenLifetime<.00001f?.00001f:minShurikenLifetime;
  2578. float initYpos = PlaygroundC.initialTargetPosition.y;
  2579. for (int p = 0; p<particleCount; p++) {
  2580. // This particle is still calculating (happens on heavier particle systems), move to next!
  2581. if (!playgroundCache.isCalculatedThisFrame[p])
  2582. continue;
  2583. // Source positions and 0 life reset
  2584. if (setSource || playgroundCache.life[p]==0f && playgroundCache.simulate[p] && !onlyLifetimePositioning) {
  2585. particleCache[p].remainingLifetime = lifetime;
  2586. playgroundCache.previousTargetPosition[p] = playgroundCache.targetPosition[p];
  2587. SetSourcePosition(p);
  2588. playgroundCache.position[p] = playgroundCache.targetPosition[p];
  2589. }
  2590. // Reposition particle
  2591. particleCache[p].position = playgroundCache.position[p];
  2592. // Shuriken lifetime (for Texture Atlas Animation)
  2593. if (playgroundCache.life[p]!=0f) {
  2594. float evalLife = playgroundCache.death[p]-localTime;
  2595. if (evalLife<mslt) evalLife = mslt;
  2596. particleCache[p].remainingLifetime = evalLife;
  2597. }
  2598. // Size
  2599. particleCache[p].size = (playgroundCache.maskAlpha[p]>0&&particleCache[p].position.y!=initYpos)?playgroundCache.size[p]:0;
  2600. // Rotation
  2601. particleCache[p].rotation = playgroundCache.rotation[p];
  2602. }
  2603. }
  2604. }
  2605. // Assign all particles into the particle system
  2606. if (!inTransition && particleCache!=null && particleCache.Length>0 && calculate && hasActiveParticles && !isPrewarming) {
  2607. shurikenParticleSystem.SetParticles(particleCache, particleCache.Length);
  2608. } else if (particleCache.Length==0 && hasActiveParticles) {
  2609. shurikenParticleSystem.Clear();
  2610. shurikenParticleSystem.Stop();
  2611. hasActiveParticles = false;
  2612. isDoneThread = true;
  2613. }
  2614. // Make sure this particle system is playing
  2615. if (hasActiveParticles && calculate)
  2616. if (shurikenParticleSystem.isPaused || shurikenParticleSystem.isStopped)
  2617. shurikenParticleSystem.Play();
  2618. }
  2619. // Initial target position
  2620. public static void SetInitialTargetPosition (PlaygroundParticlesC playgroundParticles, Vector3 position, bool refreshParticleSystem) {
  2621. for (int p = 0; p<playgroundParticles.particleCache.Length; p++) {
  2622. playgroundParticles.playgroundCache.previousTargetPosition[p] = position;
  2623. playgroundParticles.playgroundCache.targetPosition[p] = position;
  2624. playgroundParticles.particleCache[p].size = 0;
  2625. playgroundParticles.particleCache[p].position = position;
  2626. playgroundParticles.playgroundCache.position[p] = position;
  2627. playgroundParticles.playgroundCache.collisionParticlePosition[p] = position;
  2628. }
  2629. if (refreshParticleSystem)
  2630. playgroundParticles.GetComponent<ParticleSystem>().SetParticles(playgroundParticles.particleCache, playgroundParticles.particleCache.Length);
  2631. }
  2632. // Set emission of PlaygroundParticlesC object
  2633. public static void Emission (PlaygroundParticlesC playgroundParticles, bool emission, bool callRestEmission) {
  2634. playgroundParticles.previousEmission = emission;
  2635. if (emission) {
  2636. PlaygroundC.RunAsync (()=>{
  2637. for (int p = 0; p<playgroundParticles.playgroundCache.rebirth.Length; p++) {
  2638. playgroundParticles.playgroundCache.rebirth[p] = true;
  2639. if (!callRestEmission)
  2640. playgroundParticles.playgroundCache.simulate[p] = true;
  2641. }
  2642. if (callRestEmission)
  2643. SetParticleTimeNowWithRestEmission(playgroundParticles);
  2644. playgroundParticles.emissionStopped = 0;
  2645. playgroundParticles.hasActiveParticles = true;
  2646. playgroundParticles.threadHadNoActiveParticles = false;
  2647. });
  2648. }
  2649. }
  2650. // Returns the angle between a and b with normal direction
  2651. public static float SignedAngle (Vector3 a, Vector3 b, Vector3 n) {
  2652. return (Vector3.Angle(a, b)*Mathf.Sign(Vector3.Dot(n.normalized, Vector3.Cross(a, b))));
  2653. }
  2654. // Returns a random value between negative- and positive Vector3
  2655. public static Vector3 RandomVector3 (System.Random random, Vector3 v1, Vector3 v2) {
  2656. return RandomRange(random, v1, v2);
  2657. }
  2658. float realSimulationTime;
  2659. #if UNITY_EDITOR
  2660. public static bool didHierarchyRepaint = false;
  2661. #endif
  2662. /// <summary>
  2663. /// Prepares all values for calculation which is not thread-safe.
  2664. /// </summary>
  2665. /// <returns><c>true</c>, if threaded calculations was prepared, <c>false</c> otherwise.</returns>
  2666. public bool PrepareThreadedCalculations () {
  2667. // Component disabled?
  2668. if (!enabled) return false;
  2669. // Auto-Pause (start/stop calculation when particle system is out of view.
  2670. if (pauseCalculationWhenInvisible) {
  2671. bool skipFrustumCheck = false;
  2672. #if UNITY_EDITOR
  2673. if (!UnityEditor.EditorApplication.isPlaying) {
  2674. calculate = true;
  2675. skipFrustumCheck = true;
  2676. }
  2677. #endif
  2678. if (!skipFrustumCheck && Camera.main!=null && PlaygroundC.frustumPlanes!=null) {
  2679. bool shouldRender = GeometryUtility.TestPlanesAABB(PlaygroundC.frustumPlanes, new Bounds(calculationTriggerTransform==null?particleSystemTransform.position+calculationTriggerOffset:calculationTriggerTransform.position+calculationTriggerOffset, calculationTriggerSize));
  2680. if (!shouldRender && calculate && !particleSystemRenderer.isVisible) {
  2681. calculate = false;
  2682. #if UNITY_EDITOR
  2683. if (!didHierarchyRepaint)
  2684. UnityEditor.EditorApplication.RepaintHierarchyWindow();
  2685. didHierarchyRepaint = true;
  2686. #endif
  2687. return false;
  2688. } else if (shouldRender && !calculate) {
  2689. calculate = true;
  2690. #if UNITY_EDITOR
  2691. if (!didHierarchyRepaint)
  2692. UnityEditor.EditorApplication.RepaintHierarchyWindow();
  2693. didHierarchyRepaint = true;
  2694. #endif
  2695. if (prewarm) {Start(); return false;} else SetParticleTimeNow (thisInstance);
  2696. }
  2697. }
  2698. } else if (!calculate) {
  2699. calculate = true;
  2700. #if UNITY_EDITOR
  2701. if (!didHierarchyRepaint)
  2702. UnityEditor.EditorApplication.RepaintHierarchyWindow();
  2703. didHierarchyRepaint = true;
  2704. #endif
  2705. if (prewarm) {Start(); return false;} else SetParticleTimeNow (thisInstance);
  2706. }
  2707. if (!calculate) return false;
  2708. // Has the calculation been paused previously?
  2709. if (cameFromNonCalculatedFrame && hasActiveParticles) {
  2710. StartCoroutine(Boot());
  2711. cameFromNonCalculatedFrame = false;
  2712. return false;
  2713. }
  2714. // Apply locked position
  2715. if (applyLockPosition) {
  2716. if (lockPositionIsLocal)
  2717. particleSystemTransform.localPosition = lockPosition;
  2718. else
  2719. particleSystemTransform.position = lockPosition;
  2720. }
  2721. // Apply locked rotation
  2722. if (applyLockRotation) {
  2723. if (lockRotationIsLocal)
  2724. particleSystemTransform.localRotation = Quaternion.Euler(lockRotation);
  2725. else
  2726. particleSystemTransform.rotation = Quaternion.Euler(lockRotation);
  2727. }
  2728. // Apply locked scale
  2729. if (applyLockScale)
  2730. particleSystemTransform.localScale = lockScale;
  2731. // Update active particle check
  2732. hasActiveParticles = !threadHadNoActiveParticles || emit&&loop&&particleCount>0;
  2733. if (!hasActiveParticles) {
  2734. loopExceeded = true;
  2735. if (disableOnDone)
  2736. queueEmissionHalt = true;
  2737. }
  2738. realSimulationTime = particleTimescale*Time.timeScale;
  2739. cancelDeltaPositioningOnSync = false;
  2740. // Prepare Source positions
  2741. if (PlaygroundC.reference.IsDoneThread() && isDoneThread) {
  2742. stRot = Quaternion.identity;
  2743. stDir = new Vector3();
  2744. }
  2745. localSpace = (shurikenParticleSystem.simulationSpace == ParticleSystemSimulationSpace.Local);
  2746. overflow = (overflowOffset!=Vector3.zero);
  2747. skinnedWorldObjectReady = false;
  2748. renderModeStretch = particleSystemRenderer2.renderMode==ParticleSystemRenderMode.Stretch;
  2749. particleSystemRotation = particleSystemTransform.rotation;
  2750. particleSystemInverseRotation = particleSystemRotation;
  2751. particleSystemInverseRotation.x = -particleSystemInverseRotation.x;
  2752. particleSystemInverseRotation.y = -particleSystemInverseRotation.y;
  2753. particleSystemInverseRotation.z = -particleSystemInverseRotation.z;
  2754. particleSystemInverseRotation.w = -particleSystemInverseRotation.w;
  2755. // Prepare turbulence
  2756. if (!onlySourcePositioning && !onlyLifetimePositioning && turbulenceType!=TURBULENCETYPE.None) {
  2757. if (turbulenceType==TURBULENCETYPE.Simplex && turbulenceSimplex==null)
  2758. turbulenceSimplex = new SimplexNoise();
  2759. }
  2760. if (emit) {
  2761. switch (source) {
  2762. case SOURCEC.Script:
  2763. break;
  2764. case SOURCEC.State:
  2765. if (states.Count>0) {
  2766. if (activeState>states.Count-1) activeState = states.Count-1;
  2767. if (activeState<0) activeState = 0;
  2768. if (states[activeState].stateTransform!=null) {
  2769. if (isDoneThread) {
  2770. if (!states[activeState].initialized)
  2771. states[activeState].Initialize();
  2772. stRot = states[activeState].stateTransform.rotation;
  2773. states[activeState].UpdateMatrix(localSpace);
  2774. }
  2775. if (localSpace && (states[activeState].stateTransform.parent==particleSystemTransform || states[activeState].stateTransform==particleSystemTransform) && isDoneThread) {
  2776. cancelDeltaPositioningOnSync = true;
  2777. stRot = Quaternion.Euler (Vector3.zero);
  2778. }
  2779. }
  2780. } else return false;
  2781. break;
  2782. case SOURCEC.Transform:
  2783. // Handle the availability of Playground Transforms (list of transform wrapper classes).
  2784. // The single sourceTransform is an old method which needs to be copied over if available.
  2785. if (sourceTransforms==null) {
  2786. sourceTransforms = new List<PlaygroundTransformC>();
  2787. }
  2788. if (sourceTransforms.Count==0) {
  2789. sourceTransforms.Add(new PlaygroundTransformC());
  2790. sourceTransforms[0].transform = sourceTransform!=null?sourceTransform:particleSystemTransform;
  2791. }
  2792. if (isDoneThread) {
  2793. for (int i = 0; i<sourceTransforms.Count; i++) {
  2794. if (!sourceTransforms[i].Update()) return false;
  2795. if (IsLocalSpace()) {
  2796. psTransformNum = -1;
  2797. sourceTransforms[i].SetLocalPosition(particleSystemTransform);
  2798. sourceTransforms[i].SetPostitionAsLocal();
  2799. if (sourceTransforms[i].IsSameAs (particleSystemTransform)) {
  2800. sourceTransforms[i].SetZeroRotation();
  2801. psTransformNum = i;
  2802. }
  2803. }
  2804. }
  2805. }
  2806. if (isDoneThread) {
  2807. stRot = sourceTransform.rotation;
  2808. }
  2809. if (localSpace && sourceTransform==particleSystemTransform && isDoneThread) {
  2810. cancelDeltaPositioningOnSync = true;
  2811. stRot = Quaternion.Euler (Vector3.zero);
  2812. }
  2813. break;
  2814. case SOURCEC.WorldObject:
  2815. // Handle vertex data in active World Object
  2816. if (worldObject.gameObject!=null) {
  2817. if (worldObject.gameObject.GetInstanceID()!=worldObject.cachedId || !worldObject.initialized)
  2818. worldObject = NewWorldObject(worldObject.gameObject.transform);
  2819. if (worldObject.mesh!=null) {
  2820. if (isDoneThread) {
  2821. stRot = worldObject.transform.transform.rotation;
  2822. }
  2823. if (localSpace && isDoneThread) {
  2824. cancelDeltaPositioningOnSync = true;
  2825. stRot = Quaternion.Euler (Vector3.zero);
  2826. }
  2827. worldObject.UpdateMatrix(localSpace);
  2828. worldObject.updateNormals = worldObjectUpdateNormals;
  2829. if (worldObjectUpdateVertices)
  2830. worldObject.Update();
  2831. } else return false;
  2832. } else return false;
  2833. break;
  2834. case SOURCEC.SkinnedWorldObject:
  2835. // Handle vertex data in active Skinned World Object
  2836. if (skinnedWorldObject.gameObject!=null) {
  2837. if (skinnedWorldObject.gameObject.GetInstanceID()!=skinnedWorldObject.cachedId) {
  2838. skinnedWorldObject = NewSkinnedWorldObject(skinnedWorldObject.gameObject.transform, skinnedWorldObject.downResolution);
  2839. }
  2840. }
  2841. skinnedWorldObjectReady = skinnedWorldObject.gameObject!=null && skinnedWorldObject.mesh!=null;
  2842. if (PlaygroundC.reference.skinnedMeshThreadMethod==ThreadMethodComponent.OneForAll && PlaygroundC.reference.IsFirstUnsafeAutomaticFrames() && !PlaygroundC.reference.IsDoneThreadSkinnedMeshes())
  2843. return false;
  2844. if (skinnedWorldObjectReady) {
  2845. if (isDoneThread) {
  2846. stRot = skinnedWorldObject.transform.rotation;
  2847. stDir = skinnedWorldObject.transform.TransformDirection (overflowOffset);
  2848. }
  2849. skinnedWorldObject.updateNormals = worldObjectUpdateNormals;
  2850. if (Time.frameCount%PlaygroundC.skinnedUpdateRate==0) {
  2851. if (worldObjectUpdateVertices)
  2852. skinnedWorldObject.MeshUpdate();
  2853. skinnedWorldObject.BoneUpdate();
  2854. // Forced update for skinned mesh vertices on main-thread
  2855. if (forceSkinnedMeshUpdateOnMainThread) {
  2856. skinnedWorldObject.Update();
  2857. }
  2858. }
  2859. } else return false;
  2860. break;
  2861. case SOURCEC.Paint:
  2862. if (paint.initialized) {
  2863. if (isDoneThread) {
  2864. stRot = particleSystemTransform.rotation;
  2865. }
  2866. if (paint.positionLength>0) {
  2867. for (int p = 0; p<particleCache.Length; p++) {
  2868. paint.Update(p);
  2869. }
  2870. } else return false;
  2871. } else {
  2872. paint.Initialize ();
  2873. return false;
  2874. }
  2875. break;
  2876. case SOURCEC.Projection:
  2877. if (projection.projectionTexture!=null && projection.projectionTransform!=null) {
  2878. if (!projection.initialized)
  2879. projection.Initialize();
  2880. stRot = projection.projectionTransform.rotation;
  2881. if (localSpace)
  2882. shurikenParticleSystem.simulationSpace = ParticleSystemSimulationSpace.World;
  2883. if (projection.liveUpdate || !projection.hasRefreshed) {
  2884. projection.UpdateSource();
  2885. projection.Update();
  2886. stDir = projection.projectionTransform.TransformDirection (overflowOffset);
  2887. projection.hasRefreshed = true;
  2888. }
  2889. } else return false;
  2890. break;
  2891. case SOURCEC.Spline:
  2892. if (splines.Count>0) {
  2893. for (int i = 0; i<splines.Count; i++)
  2894. if (splines[i]==null || !splines[i].enabled || !splines[i].IsReady()) return false;
  2895. if (isDoneThread) {
  2896. stRot = particleSystemTransform.rotation;
  2897. if (localSpace) {
  2898. cancelDeltaPositioningOnSync = true;
  2899. stRot = Quaternion.Euler (Vector3.zero);
  2900. }
  2901. }
  2902. } else return false;
  2903. break;
  2904. }
  2905. }
  2906. // Prepare Particle colors
  2907. stateReadyForTextureColor = (source==SOURCEC.State && states[activeState].stateTexture!=null && states[activeState].colorLength>0);
  2908. hasEventManipulatorLocal = false;
  2909. hasEventManipulatorGlobal = false;
  2910. hasGlobalAffectingManipulators = false;
  2911. manipulatorEventCount = 0;
  2912. thisLayer = particleSystemGameObject.layer;
  2913. // Prepare Local Manipulators
  2914. for (int m = 0; m<manipulators.Count; m++) {
  2915. manipulators[m].Update();
  2916. manipulators[m].transform.SetLocalPosition(particleSystemTransform);
  2917. manipulators[m].SetLocalTargetsPosition(particleSystemTransform);
  2918. if (manipulators[m].trackParticles) {
  2919. hasEventManipulatorLocal = true;
  2920. manipulatorEventCount++;
  2921. }
  2922. }
  2923. // Prepare Global Manipulators from this local space, bundle the thread call in case any Global Manipulator is tracking particles
  2924. if (PlaygroundC.reference.HasEnabledGlobalManipulators()) {
  2925. for (int m = 0; m<PlaygroundC.reference.manipulators.Count; m++) {
  2926. if ((PlaygroundC.reference.manipulators[m].affects.value & 1<<GetLayer())!=0) {
  2927. hasGlobalAffectingManipulators = true;
  2928. PlaygroundC.reference.manipulators[m].transform.SetLocalPosition(particleSystemTransform);
  2929. if (PlaygroundC.reference.manipulators[m].trackParticles && PlaygroundC.reference.threadMethod!=ThreadMethod.NoThreads) {
  2930. threadMethod = ThreadMethodLocal.OneForAll;
  2931. hasEventManipulatorGlobal = true;
  2932. manipulatorEventCount++;
  2933. }
  2934. }
  2935. }
  2936. }
  2937. hasSeveralManipulatorEvents = (manipulatorEventCount>1);
  2938. // Prepare events
  2939. hasEvent = events.Count>0;
  2940. hasTimerEvent = false;
  2941. if (hasEvent) {
  2942. for (int i = 0; i<events.Count; i++) {
  2943. events[i].Initialize();
  2944. if (events[i].initializedTarget && events[i].broadcastType!=EVENTBROADCASTC.EventListeners)
  2945. if (!events[i].target.eventControlledBy.Contains (thisInstance))
  2946. events[i].target.eventControlledBy.Add (thisInstance);
  2947. if (events[i].eventType==EVENTTYPEC.Time && ((events[i].broadcastType!=EVENTBROADCASTC.EventListeners && events[i].initializedTarget) || events[i].broadcastType==EVENTBROADCASTC.EventListeners && events[i].initializedEvent)) {
  2948. hasTimerEvent = events[i].UpdateTime();
  2949. }
  2950. }
  2951. }
  2952. // Prepare collision cache checking
  2953. hasCollisionCache = collisionCache!=null;
  2954. // All good! Next step is to send all particles to calculation.
  2955. return true;
  2956. }
  2957. public static void NewCalculatedThread (PlaygroundParticlesC playgroundParticles) {
  2958. if (playgroundParticles.isDoneThread && playgroundParticles.isReadyForThreadedCalculations && !playgroundParticles.IsYieldRefreshing()) {
  2959. playgroundParticles.isDoneThread = false;
  2960. PlaygroundC.RunAsync(()=> {
  2961. if (playgroundParticles.isDoneThread || !playgroundParticles.isReadyForThreadedCalculations) return;
  2962. ThreadedCalculations(playgroundParticles);
  2963. });
  2964. }
  2965. }
  2966. public static void NewCalculatedThread (PlaygroundParticlesC[] playgroundParticles) {
  2967. for (int i = 0; i<playgroundParticles.Length; i++) {
  2968. if (playgroundParticles[i].isDoneThread && !playgroundParticles[i].IsYieldRefreshing() && playgroundParticles[i].threadMethod==ThreadMethodLocal.Inherit)
  2969. playgroundParticles[i].isDoneThread = false;
  2970. }
  2971. PlaygroundC.RunAsync(()=> {
  2972. for (int i = 0; i<playgroundParticles.Length; i++) {
  2973. if (!playgroundParticles[i].isDoneThread && PlaygroundC.reference.IsDoneThread() && playgroundParticles[i].isReadyForThreadedCalculations && playgroundParticles[i].threadMethod==ThreadMethodLocal.Inherit) {
  2974. ThreadedCalculations(playgroundParticles[i]);
  2975. }
  2976. }
  2977. });
  2978. }
  2979. Color32 GetParticleColor (int p, float normalizedLife, float normalizedP) {
  2980. Color currentColor = new Color();
  2981. // Get this particle's color
  2982. if (!playgroundCache.changedByPropertyColor[p] && !playgroundCache.changedByPropertyColorLerp[p]) {
  2983. switch (colorSource) {
  2984. case COLORSOURCEC.Source:
  2985. switch (source) {
  2986. case SOURCEC.Script:
  2987. currentColor = playgroundCache.scriptedColor[p];
  2988. break;
  2989. case SOURCEC.State:
  2990. if (stateReadyForTextureColor)
  2991. currentColor = states[activeState].GetColor(p%states[activeState].colorLength);
  2992. else
  2993. currentColor = lifetimeColor.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP);
  2994. break;
  2995. case SOURCEC.Projection:
  2996. currentColor = projection.GetColor(p);
  2997. break;
  2998. case SOURCEC.Paint:
  2999. currentColor = paint.GetColor(p);
  3000. break;
  3001. default:
  3002. currentColor = lifetimeColor.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP);
  3003. break;
  3004. }
  3005. if (sourceUsesLifetimeAlpha)
  3006. currentColor.a = Mathf.Clamp(lifetimeColor.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP).a, 0, currentColor.a);
  3007. break;
  3008. case COLORSOURCEC.LifetimeColor:
  3009. currentColor = lifetimeColor.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP);
  3010. break;
  3011. case COLORSOURCEC.LifetimeColors:
  3012. if (lifetimeColors.Count>0)
  3013. currentColor = lifetimeColors[playgroundCache.lifetimeColorId[p]%lifetimeColors.Count].gradient.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP);
  3014. else
  3015. currentColor = lifetimeColor.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP);
  3016. break;
  3017. }
  3018. } else {
  3019. currentColor = particleCache[p].color;
  3020. if (playgroundCache.changedByPropertyColorKeepAlpha[p])
  3021. currentColor.a = Mathf.Clamp(lifetimeColor.Evaluate(colorMethod==COLORMETHOD.Lifetime?normalizedLife:normalizedP).a, 0, currentColor.a);
  3022. }
  3023. if (colorMethod==COLORMETHOD.ParticleArray && arrayColorUsesAlpha) {
  3024. Color arrayAlphaColor;
  3025. arrayAlphaColor = arrayColorAlpha.Evaluate(normalizedLife);
  3026. currentColor.a = arrayAlphaColor.a;
  3027. }
  3028. // Assign mask alpha
  3029. if (applyParticleMask) {
  3030. currentColor.a*=playgroundCache.maskAlpha[p];
  3031. }
  3032. return currentColor;
  3033. }
  3034. /// <summary>
  3035. /// Sets the source position for a particle from the selected source method in the particle system.
  3036. /// </summary>
  3037. /// <param name="p">Particle Index.</param>
  3038. void SetSourcePosition (int p) {
  3039. switch (source) {
  3040. case SOURCEC.State:
  3041. if (!overflow) {
  3042. playgroundCache.targetPosition[p] = states[activeState].GetLocalPosition(p);
  3043. } else {
  3044. switch (overflowMode) {
  3045. case OVERFLOWMODEC.SourceTransform:
  3046. playgroundCache.targetPosition[p] = states[activeState].GetLocalPosition(p)+GetOverflow2(overflowOffset, p, states[activeState].positionLength);
  3047. break;
  3048. case OVERFLOWMODEC.World:
  3049. playgroundCache.targetPosition[p] = states[activeState].GetLocalPosition(p)+GetOverflow2(overflowOffset, p, states[activeState].positionLength);
  3050. break;
  3051. case OVERFLOWMODEC.SourcePoint:
  3052. playgroundCache.targetPosition[p] = states[activeState].GetLocalPosition(p)+GetOverflow2(overflowOffset, states[activeState].GetLocalNormal(p), p, states[activeState].positionLength);
  3053. break;
  3054. }
  3055. }
  3056. break;
  3057. case SOURCEC.Transform:
  3058. if (sourceTransforms.Count>0) {
  3059. if (treatAsOneTransform)
  3060. transformIndex = (int)((((p*1f)/(particleCount*1f))*sourceTransforms.Count))%sourceTransforms.Count;
  3061. else
  3062. transformIndex = p%sourceTransforms.Count;
  3063. if (transformIndex>=sourceTransforms.Count) {
  3064. transformIndex=sourceTransforms.Count-1;
  3065. }
  3066. if (!overflow) {
  3067. playgroundCache.targetPosition[p] = sourceTransforms[transformIndex].position;
  3068. } else {
  3069. switch (overflowMode) {
  3070. case OVERFLOWMODEC.SourceTransform:
  3071. playgroundCache.targetPosition[p] = sourceTransforms[transformIndex].transformMatrix.MultiplyPoint3x4(GetOverflow2(overflowOffset, treatAsOneTransform? p%(particleCount/sourceTransforms.Count) : p/sourceTransforms.Count, 1));
  3072. break;
  3073. case OVERFLOWMODEC.World:
  3074. playgroundCache.targetPosition[p] = sourceTransforms[transformIndex].position+GetOverflow2(overflowOffset, treatAsOneTransform? p%(particleCount/sourceTransforms.Count) : p/sourceTransforms.Count, 1);
  3075. break;
  3076. case OVERFLOWMODEC.SourcePoint:
  3077. playgroundCache.targetPosition[p] = sourceTransforms[transformIndex].transformMatrix.MultiplyPoint3x4(GetOverflow2(overflowOffset, treatAsOneTransform? p%(particleCount/sourceTransforms.Count) : p/sourceTransforms.Count, 1));
  3078. break;
  3079. }
  3080. }
  3081. }
  3082. break;
  3083. case SOURCEC.WorldObject:
  3084. if (!overflow) {
  3085. playgroundCache.targetPosition[p] = worldObject.transformMatrix.MultiplyPoint3x4(
  3086. worldObject.vertexPositions[p%worldObject.vertexPositions.Length]
  3087. );
  3088. } else {
  3089. switch (overflowMode) {
  3090. case OVERFLOWMODEC.SourceTransform:
  3091. playgroundCache.targetPosition[p] = worldObject.transformMatrix.MultiplyPoint3x4(
  3092. worldObject.vertexPositions[p%worldObject.vertexPositions.Length]+GetOverflow2(overflowOffset, p, worldObject.vertexPositions.Length)
  3093. );
  3094. break;
  3095. case OVERFLOWMODEC.World:
  3096. playgroundCache.targetPosition[p] = worldObject.transformMatrix.MultiplyPoint3x4(
  3097. worldObject.vertexPositions[p%worldObject.vertexPositions.Length]
  3098. )+GetOverflow2(overflowOffset, p, worldObject.vertexPositions.Length);
  3099. break;
  3100. case OVERFLOWMODEC.SourcePoint:
  3101. playgroundCache.targetPosition[p] = worldObject.transformMatrix.MultiplyPoint3x4(
  3102. worldObject.vertexPositions[p%worldObject.vertexPositions.Length]+GetOverflow2(overflowOffset, worldObject.normals[p%worldObject.normals.Length], p, worldObject.vertexPositions.Length)
  3103. );
  3104. break;
  3105. }
  3106. }
  3107. break;
  3108. case SOURCEC.SkinnedWorldObject:
  3109. int downResolution = skinnedWorldObject.downResolution;
  3110. int downResolutionP = Mathf.RoundToInt(p*downResolution)%skinnedWorldObject.vertexPositions.Length;
  3111. if (!overflow) {
  3112. playgroundCache.targetPosition[p] = skinnedWorldObject.vertexPositions[downResolutionP];
  3113. } else {
  3114. switch (overflowMode) {
  3115. case OVERFLOWMODEC.SourceTransform:
  3116. playgroundCache.targetPosition[p] = skinnedWorldObject.vertexPositions[downResolutionP]+
  3117. GetOverflow2(
  3118. stDir,
  3119. p,
  3120. skinnedWorldObject.vertexPositions.Length/downResolution
  3121. );
  3122. break;
  3123. case OVERFLOWMODEC.World:
  3124. playgroundCache.targetPosition[p] = skinnedWorldObject.vertexPositions[downResolutionP]+
  3125. GetOverflow2(
  3126. overflowOffset,
  3127. p,
  3128. skinnedWorldObject.vertexPositions.Length/downResolution
  3129. );
  3130. break;
  3131. case OVERFLOWMODEC.SourcePoint:
  3132. playgroundCache.targetPosition[p] = skinnedWorldObject.vertexPositions[downResolutionP]+
  3133. GetOverflow2(
  3134. overflowOffset,
  3135. skinnedWorldObject.normals[downResolutionP],
  3136. p,
  3137. skinnedWorldObject.vertexPositions.Length/downResolution
  3138. );
  3139. break;
  3140. }
  3141. }
  3142. break;
  3143. case SOURCEC.Projection:
  3144. if (!overflow) {
  3145. playgroundCache.targetPosition[p] = projection.GetPosition(p);
  3146. } else {
  3147. switch (overflowMode) {
  3148. case OVERFLOWMODEC.SourceTransform:
  3149. playgroundCache.targetPosition[p] = projection.GetPosition(p)+GetOverflow2(stDir, p, projection.positionLength);
  3150. break;
  3151. case OVERFLOWMODEC.World:
  3152. playgroundCache.targetPosition[p] = projection.GetPosition(p)+GetOverflow2(overflowOffset, p, projection.positionLength);
  3153. break;
  3154. case OVERFLOWMODEC.SourcePoint:
  3155. playgroundCache.targetPosition[p] = projection.GetPosition(p)+GetOverflow2(Vector3Scale(stDir, projection.GetNormal(p)), p, projection.positionLength);
  3156. break;
  3157. }
  3158. }
  3159. break;
  3160. case SOURCEC.Paint:
  3161. if (!overflow) {
  3162. playgroundCache.targetPosition[p] = paint.GetPosition(p);
  3163. } else {
  3164. switch (overflowMode) {
  3165. case OVERFLOWMODEC.SourceTransform:
  3166. playgroundCache.targetPosition[p] = paint.GetPosition(p)+GetOverflow2(paint.GetRotation(p)*overflowOffset, p, paint.positionLength);
  3167. break;
  3168. case OVERFLOWMODEC.World:
  3169. playgroundCache.targetPosition[p] = paint.GetPosition(p)+GetOverflow2(overflowOffset, p, paint.positionLength);
  3170. break;
  3171. case OVERFLOWMODEC.SourcePoint:
  3172. playgroundCache.targetPosition[p] = paint.GetPosition(p)+paint.GetRotation(p)*GetOverflow2(overflowOffset, paint.GetNormal(p), p, paint.positionLength);
  3173. break;
  3174. }
  3175. }
  3176. break;
  3177. case SOURCEC.Spline:
  3178. int sIndex;
  3179. if (treatAsOneSpline)
  3180. sIndex = (int)((((p*1f)/(particleCount*1f))*splines.Count)+splineTimeOffset)%splines.Count;
  3181. else
  3182. sIndex = p%splines.Count;
  3183. float pSplineTime;
  3184. if (treatAsOneSpline)
  3185. pSplineTime = ((p*splines.Count*1f) / (particleCount*1f))+splineTimeOffset;
  3186. else
  3187. pSplineTime = ((p*1f) / (particleCount*1f)) + splineTimeOffset;
  3188. playgroundCache.targetPosition[p] = splines[sIndex].GetPoint(pSplineTime);
  3189. break;
  3190. }
  3191. if (applySourceScatter)
  3192. playgroundCache.targetPosition[p] += Vector3.Scale (playgroundCache.scatterPosition[p], scatterScale);
  3193. }
  3194. /// <summary>
  3195. /// Thread-safe particle calculations.
  3196. /// </summary>
  3197. private bool isDoneThread = true;
  3198. bool threadHadNoActiveParticles = false;
  3199. public static void ThreadedCalculations (PlaygroundParticlesC playgroundParticles) {
  3200. // Refresh delta time
  3201. if (!playgroundParticles.isPrewarming) {
  3202. playgroundParticles.localDeltaTime = (PlaygroundC.globalTime-playgroundParticles.lastTimeUpdated)*playgroundParticles.particleTimescale;
  3203. playgroundParticles.localTime += playgroundParticles.localDeltaTime;
  3204. playgroundParticles.lastTimeUpdated = PlaygroundC.globalTime;
  3205. }
  3206. // Set delta time
  3207. playgroundParticles.t = playgroundParticles.localDeltaTime;
  3208. if (playgroundParticles.particleCount<=0 ||
  3209. playgroundParticles.playgroundCache.color.Length!=playgroundParticles.particleCount ||
  3210. playgroundParticles.playgroundCache.targetPosition.Length!=playgroundParticles.particleCount ||
  3211. playgroundParticles.playgroundCache.targetDirection.Length!=playgroundParticles.particleCount ||
  3212. playgroundParticles.source==SOURCEC.State && playgroundParticles.states[playgroundParticles.activeState].IsInitializing() ||
  3213. playgroundParticles.isYieldRefreshing || !PlaygroundC.IsReady() || playgroundParticles.isSettingParticleTime) {
  3214. playgroundParticles.isDoneThread = true;
  3215. playgroundParticles.threadHadNoActiveParticles = false;
  3216. return;
  3217. }
  3218. if (PlaygroundC.reference.calculate && playgroundParticles.calculate && !playgroundParticles.inTransition && playgroundParticles.hasActiveParticles) {}
  3219. else if (playgroundParticles.source!=SOURCEC.Script) {
  3220. playgroundParticles.isDoneThread = true;
  3221. playgroundParticles.cameFromNonCalculatedFrame = true;
  3222. return;
  3223. } else {
  3224. playgroundParticles.isDoneThread = true;
  3225. return;
  3226. }
  3227. // Check that simplex turbulence is available (will be next frame in case not)
  3228. if (!playgroundParticles.onlySourcePositioning && !playgroundParticles.onlyLifetimePositioning && playgroundParticles.turbulenceStrength>0 && playgroundParticles.turbulenceType!=TURBULENCETYPE.None) {
  3229. if (playgroundParticles.turbulenceType==TURBULENCETYPE.Simplex && playgroundParticles.turbulenceSimplex==null) {
  3230. playgroundParticles.isDoneThread = true;
  3231. playgroundParticles.threadHadNoActiveParticles = false;
  3232. return;
  3233. }
  3234. }
  3235. float t = playgroundParticles.t;
  3236. // Prepare variables for particle source positions
  3237. Matrix4x4 fMx = new Matrix4x4();
  3238. if (playgroundParticles.source==SOURCEC.State || playgroundParticles.source==SOURCEC.WorldObject || playgroundParticles.source==SOURCEC.SkinnedWorldObject) {
  3239. fMx.SetTRS(Vector3.zero, playgroundParticles.stRot, new Vector3(1f,1f,1f));
  3240. }
  3241. bool noActiveParticles = true;
  3242. if (playgroundParticles.source==SOURCEC.Transform)
  3243. for (int i = 0; i<playgroundParticles.sourceTransforms.Count; i++)
  3244. playgroundParticles.sourceTransforms[i].UpdateMatrix();
  3245. // Update skinned mesh vertices
  3246. if (playgroundParticles.source==SOURCEC.SkinnedWorldObject && playgroundParticles.skinnedWorldObjectReady && !playgroundParticles.forceSkinnedMeshUpdateOnMainThread && PlaygroundC.reference.skinnedMeshThreadMethod==ThreadMethodComponent.InsideParticleCalculation) {
  3247. playgroundParticles.skinnedWorldObject.Update();
  3248. }
  3249. // Misc
  3250. int pCount = playgroundParticles.particleCache.Length;
  3251. // Check that cache is correct
  3252. if (playgroundParticles.playgroundCache.lifetimeSubtraction.Length!=pCount)
  3253. SetLifetimeSubtraction(playgroundParticles);
  3254. if (playgroundParticles.playgroundCache.maskAlpha.Length!=pCount) {
  3255. playgroundParticles.playgroundCache.maskAlpha = new float[pCount];
  3256. playgroundParticles.playgroundCache.isMasked = new bool[pCount];
  3257. }
  3258. if (playgroundParticles.playgroundCache.manipulatorId.Length!=pCount)
  3259. playgroundParticles.playgroundCache.manipulatorId = new int[pCount];
  3260. if (playgroundParticles.playgroundCache.excludeFromManipulatorId.Length!=pCount)
  3261. playgroundParticles.playgroundCache.excludeFromManipulatorId = new int[pCount];
  3262. if (playgroundParticles.playgroundCache.noForce.Length!=pCount)
  3263. playgroundParticles.playgroundCache.noForce = new bool[pCount];
  3264. if (playgroundParticles.playgroundCache.isNonBirthed.Length!=pCount)
  3265. playgroundParticles.playgroundCache.isNonBirthed = new bool[pCount];
  3266. if (playgroundParticles.playgroundCache.isFirstLoop.Length!=pCount)
  3267. playgroundParticles.playgroundCache.isFirstLoop = new bool[pCount];
  3268. if (playgroundParticles.playgroundCache.isCalculatedThisFrame.Length!=pCount)
  3269. playgroundParticles.playgroundCache.isCalculatedThisFrame = new bool[pCount];
  3270. if (playgroundParticles.playgroundCache.simulate.Length!=pCount) {
  3271. playgroundParticles.playgroundCache.simulate = new bool[pCount];
  3272. for (int p = 0; p<pCount; p++)
  3273. playgroundParticles.playgroundCache.simulate[p] = true;
  3274. }
  3275. if (playgroundParticles.applyParticleMask && (playgroundParticles.playgroundCache.maskSorting==null || playgroundParticles.playgroundCache.maskSorting.Length!=pCount || playgroundParticles.particleMaskSorting!=playgroundParticles.previousMaskSorting))
  3276. playgroundParticles.RefreshMaskSorting();
  3277. Vector3 zero = Vector3.zero;
  3278. Vector3 up = Vector3.up;
  3279. float initYpos = PlaygroundC.initialTargetPosition.y;
  3280. // Calculation loop
  3281. for (int p = 0; p<playgroundParticles.particleCount; p++) {
  3282. // Check that particle count is correct
  3283. if (pCount != playgroundParticles.particleCache.Length || playgroundParticles.isYieldRefreshing || playgroundParticles.isSettingParticleTime || playgroundParticles.isSettingParticleCount) {
  3284. playgroundParticles.cameFromNonEmissionFrame = false;
  3285. playgroundParticles.isDoneThread = true;
  3286. playgroundParticles.threadHadNoActiveParticles = false;
  3287. return;
  3288. }
  3289. // Check simulation
  3290. if (!playgroundParticles.playgroundCache.simulate[p]) {
  3291. if (playgroundParticles.playgroundCache.rebirth[p] && playgroundParticles.loop) {
  3292. playgroundParticles.InactivateParticle(p);
  3293. }
  3294. continue;
  3295. } else {
  3296. noActiveParticles = false;
  3297. }
  3298. // This particle is about to be calculated
  3299. playgroundParticles.playgroundCache.isCalculatedThisFrame[p] = false;
  3300. // Prepare variables inside scope
  3301. float lifeInSeconds = (playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p])-playgroundParticles.playgroundCache.lifetimeSubtraction[p];
  3302. float normalizedLife = Mathf.Clamp01(playgroundParticles.playgroundCache.life[p]/lifeInSeconds);
  3303. float normalizedP = (p*1f)/(playgroundParticles.particleCount*1f);
  3304. float lifetimePositioningTimeScale = 1f;
  3305. if (playgroundParticles.applyLifetimePositioningTimeScale)
  3306. lifetimePositioningTimeScale = playgroundParticles.lifetimePositioningTimeScale.Evaluate(normalizedLife);
  3307. // Apply particle mask
  3308. if (playgroundParticles.applyParticleMask) {
  3309. int maskedP = playgroundParticles.playgroundCache.maskSorting[p];
  3310. if (p<playgroundParticles.particleMask) {
  3311. if (playgroundParticles.playgroundCache.maskAlpha[maskedP]<=0 || playgroundParticles.particleMaskTime<=0) {
  3312. playgroundParticles.playgroundCache.isMasked[maskedP] = true;
  3313. playgroundParticles.playgroundCache.maskAlpha[maskedP] = 0;
  3314. playgroundParticles.particleCache[maskedP].size = 0;
  3315. } else {
  3316. playgroundParticles.playgroundCache.maskAlpha[maskedP] -= (1f/playgroundParticles.particleMaskTime)*playgroundParticles.localDeltaTime;
  3317. }
  3318. } else {
  3319. if (playgroundParticles.playgroundCache.maskAlpha[maskedP]>=1f || playgroundParticles.particleMaskTime<=0) {
  3320. playgroundParticles.playgroundCache.isMasked[maskedP] = false;
  3321. playgroundParticles.playgroundCache.maskAlpha[maskedP] = 1f;
  3322. } else {
  3323. playgroundParticles.playgroundCache.maskAlpha[maskedP] += (1f/playgroundParticles.particleMaskTime)*playgroundParticles.localDeltaTime;
  3324. }
  3325. }
  3326. } else {
  3327. playgroundParticles.playgroundCache.isMasked[p] = false;
  3328. playgroundParticles.playgroundCache.maskAlpha[p] = 1f;
  3329. }
  3330. Color32 lifetimeColor = playgroundParticles.GetParticleColor(p, normalizedLife, normalizedP);
  3331. // Assign color to particle
  3332. playgroundParticles.particleCache[p].color = lifetimeColor;
  3333. // Give Playground Cache its color value
  3334. playgroundParticles.playgroundCache.color[p] = lifetimeColor;
  3335. // Source positions
  3336. if (playgroundParticles.emit) {
  3337. if (!playgroundParticles.syncPositionsOnMainThread)
  3338. playgroundParticles.playgroundCache.previousTargetPosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  3339. switch (playgroundParticles.source) {
  3340. case SOURCEC.State:
  3341. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3342. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3343. playgroundParticles.playgroundCache.targetDirection[p] = fMx.MultiplyPoint3x4(Vector3Scale(playgroundParticles.playgroundCache.initialLocalVelocity[p], playgroundParticles.states[playgroundParticles.activeState].GetNormal(p)));
  3344. if (playgroundParticles.onlyLifetimePositioning) {
  3345. playgroundParticles.playgroundCache.targetDirection[p] = fMx.MultiplyPoint3x4(Vector3Scale(playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale), playgroundParticles.states[playgroundParticles.activeState].GetNormal(p)));
  3346. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3347. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife);
  3348. }
  3349. playgroundParticles.SetSourcePosition(p);
  3350. // Local space compensation calculation
  3351. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation)
  3352. playgroundParticles.playgroundCache.localSpaceMovementCompensation[p] = playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p];
  3353. }
  3354. break;
  3355. case SOURCEC.Transform:
  3356. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3357. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3358. if (playgroundParticles.localSpace && playgroundParticles.transformIndex!=playgroundParticles.psTransformNum)
  3359. playgroundParticles.playgroundCache.targetDirection[p] = playgroundParticles.particleSystemInverseRotation*playgroundParticles.sourceTransforms[playgroundParticles.transformIndex].rotation*playgroundParticles.playgroundCache.initialLocalVelocity[p];
  3360. else
  3361. playgroundParticles.playgroundCache.targetDirection[p] = playgroundParticles.sourceTransforms[playgroundParticles.transformIndex].rotation*playgroundParticles.playgroundCache.initialLocalVelocity[p];
  3362. if (playgroundParticles.onlyLifetimePositioning) {
  3363. playgroundParticles.playgroundCache.targetDirection[p] = playgroundParticles.sourceTransforms[playgroundParticles.transformIndex].rotation*playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale);
  3364. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3365. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife*lifetimePositioningTimeScale);
  3366. }
  3367. playgroundParticles.SetSourcePosition(p);
  3368. // Local space compensation calculation
  3369. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation)
  3370. playgroundParticles.playgroundCache.localSpaceMovementCompensation[p] = playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p];
  3371. }
  3372. break;
  3373. case SOURCEC.WorldObject:
  3374. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3375. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3376. playgroundParticles.playgroundCache.targetDirection[p] = fMx.MultiplyPoint3x4(Vector3Scale(playgroundParticles.playgroundCache.initialLocalVelocity[p], playgroundParticles.worldObject.normals[p%playgroundParticles.worldObject.normals.Length]));
  3377. if (playgroundParticles.onlyLifetimePositioning) {
  3378. playgroundParticles.playgroundCache.targetDirection[p] = fMx.MultiplyPoint3x4(Vector3Scale(playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale), playgroundParticles.worldObject.normals[p%playgroundParticles.worldObject.normals.Length]));
  3379. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3380. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife);
  3381. }
  3382. playgroundParticles.SetSourcePosition(p);
  3383. // Local space compensation calculation
  3384. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation)
  3385. playgroundParticles.playgroundCache.localSpaceMovementCompensation[p] = (playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p]);
  3386. }
  3387. break;
  3388. case SOURCEC.SkinnedWorldObject:
  3389. if (playgroundParticles.skinnedWorldObjectReady) {
  3390. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3391. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3392. playgroundParticles.playgroundCache.targetDirection[p] = fMx.MultiplyPoint3x4(Vector3Scale(playgroundParticles.playgroundCache.initialLocalVelocity[p], playgroundParticles.skinnedWorldObject.normals[p%playgroundParticles.skinnedWorldObject.normals.Length]));
  3393. if (playgroundParticles.onlyLifetimePositioning) {
  3394. playgroundParticles.playgroundCache.targetDirection[p] = fMx.MultiplyPoint3x4(Vector3Scale(playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale), playgroundParticles.skinnedWorldObject.normals[p%playgroundParticles.skinnedWorldObject.normals.Length]));
  3395. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3396. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife);
  3397. }
  3398. playgroundParticles.SetSourcePosition(p);
  3399. // Local space compensation calculation
  3400. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation)
  3401. playgroundParticles.playgroundCache.localSpaceMovementCompensation[p] = playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p];
  3402. }
  3403. }
  3404. break;
  3405. case SOURCEC.Projection:
  3406. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3407. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3408. playgroundParticles.playgroundCache.targetDirection[p] = Vector3Scale(playgroundParticles.projection.GetNormal(p), playgroundParticles.playgroundCache.initialLocalVelocity[p]);
  3409. if (playgroundParticles.onlyLifetimePositioning) {
  3410. playgroundParticles.playgroundCache.targetDirection[p] = Vector3Scale(playgroundParticles.projection.GetNormal(p), playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale));
  3411. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3412. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife);
  3413. }
  3414. playgroundParticles.SetSourcePosition(p);
  3415. }
  3416. break;
  3417. case SOURCEC.Paint:
  3418. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3419. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3420. playgroundParticles.playgroundCache.targetDirection[p] = playgroundParticles.paint.GetRotation(p)*Vector3Scale(playgroundParticles.playgroundCache.initialLocalVelocity[p], playgroundParticles.paint.GetNormal(p));
  3421. if (playgroundParticles.onlyLifetimePositioning) {
  3422. playgroundParticles.playgroundCache.targetDirection[p] = playgroundParticles.paint.GetRotation(p)*Vector3Scale(playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale), playgroundParticles.paint.GetNormal(p));
  3423. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3424. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife);
  3425. }
  3426. playgroundParticles.SetSourcePosition(p);
  3427. // Local space compensation calculation
  3428. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation)
  3429. playgroundParticles.playgroundCache.localSpaceMovementCompensation[p] = playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p];
  3430. }
  3431. break;
  3432. case SOURCEC.Spline:
  3433. if (playgroundParticles.playgroundCache.rebirth[p] || playgroundParticles.onlySourcePositioning || playgroundParticles.onlyLifetimePositioning) {
  3434. if (playgroundParticles.treatAsOneSpline)
  3435. playgroundParticles.splineIndex = (int)((((p*1f)/(pCount*1f))*playgroundParticles.splines.Count)+playgroundParticles.splineTimeOffset)%playgroundParticles.splines.Count;
  3436. else
  3437. playgroundParticles.splineIndex = p%playgroundParticles.splines.Count;
  3438. float pSplineTime;
  3439. if (playgroundParticles.treatAsOneSpline)
  3440. pSplineTime = ((p*playgroundParticles.splines.Count*1f) / (pCount*1f))+playgroundParticles.splineTimeOffset;
  3441. else
  3442. pSplineTime = ((p*1f) / (pCount*1f)) + playgroundParticles.splineTimeOffset;
  3443. if (playgroundParticles.applyInitialLocalVelocity && !playgroundParticles.onlyLifetimePositioning)
  3444. playgroundParticles.playgroundCache.targetDirection[p] = Vector3Scale(playgroundParticles.splines[playgroundParticles.splineIndex].GetDirection(pSplineTime), playgroundParticles.playgroundCache.initialLocalVelocity[p]);
  3445. if (playgroundParticles.onlyLifetimePositioning) {
  3446. playgroundParticles.playgroundCache.targetDirection[p] = Vector3Scale(playgroundParticles.splines[playgroundParticles.splineIndex].GetDirection(pSplineTime), playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale));
  3447. if (playgroundParticles.applyLifetimePositioningPositionScale)
  3448. playgroundParticles.playgroundCache.targetDirection[p] *= playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife*lifetimePositioningTimeScale);
  3449. }
  3450. playgroundParticles.playgroundCache.targetPosition[p] = playgroundParticles.splines[playgroundParticles.splineIndex].GetPoint(pSplineTime);
  3451. if (playgroundParticles.applySourceScatter)
  3452. playgroundParticles.playgroundCache.targetPosition[p] += Vector3.Scale (playgroundParticles.playgroundCache.scatterPosition[p],playgroundParticles.scatterScale);
  3453. // Local space compensation calculation
  3454. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation)
  3455. playgroundParticles.playgroundCache.localSpaceMovementCompensation[p] = playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p];
  3456. }
  3457. break;
  3458. }
  3459. }
  3460. // Set initial particle values if life is 0
  3461. if (playgroundParticles.playgroundCache.life[p]==0) {
  3462. if (!playgroundParticles.onlyLifetimePositioning) {
  3463. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.targetPosition[p];
  3464. // Apply birth position delta
  3465. if (playgroundParticles.applyDeltaOnRebirth && !playgroundParticles.cameFromNonEmissionFrame && playgroundParticles.localTime-playgroundParticles.emissionStopped > playgroundParticles.lifetime && playgroundParticles.playgroundCache.birthDelay[p]==0 && !playgroundParticles.onlySourcePositioning) {
  3466. float timeDelta = playgroundParticles.localTime-playgroundParticles.playgroundCache.birth[p];
  3467. if (playgroundParticles.isPrewarming) timeDelta *=.5f;
  3468. playgroundParticles.playgroundCache.position[p] += playgroundParticles.playgroundCache.velocity[p]*playgroundParticles.velocityScale*timeDelta;
  3469. }
  3470. }
  3471. playgroundParticles.playgroundCache.initialColor[p] = lifetimeColor;
  3472. // Delta movement velocity
  3473. if (!playgroundParticles.cameFromNonEmissionFrame && playgroundParticles.calculateDeltaMovement && !playgroundParticles.isPrewarming && playgroundParticles.source!=SOURCEC.Script) {
  3474. Vector3 deltaVelocity = playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.previousTargetPosition[p];
  3475. if (playgroundParticles.deltaMovementStrengthValueMethod == VALUEMETHOD.Constant)
  3476. playgroundParticles.playgroundCache.velocity[p] += (deltaVelocity*playgroundParticles.deltaMovementStrength)/playgroundParticles.realSimulationTime;
  3477. else
  3478. playgroundParticles.playgroundCache.velocity[p] += (deltaVelocity*RandomRange(playgroundParticles.internalRandom01, playgroundParticles.deltaMovementStrength, playgroundParticles.minDeltaMovementStrength))/playgroundParticles.realSimulationTime;
  3479. }
  3480. playgroundParticles.playgroundCache.birthDelay[p] = 0;
  3481. }
  3482. if (playgroundParticles.playgroundCache.birth[p]>playgroundParticles.localTime) {
  3483. playgroundParticles.particleCache[p].size = 0;
  3484. playgroundParticles.playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  3485. }
  3486. // Particle lifetime, velocity and manipulators
  3487. if (playgroundParticles.playgroundCache.rebirth[p]) {
  3488. // Particle is alive
  3489. if (playgroundParticles.playgroundCache.birth[p]+playgroundParticles.playgroundCache.life[p]<=playgroundParticles.localTime+lifeInSeconds && (!playgroundParticles.playgroundCache.isNonBirthed[p] || playgroundParticles.onlyLifetimePositioning || playgroundParticles.onlySourcePositioning)) {
  3490. // Lifetime size
  3491. if (!playgroundParticles.playgroundCache.changedByPropertySize[p]) {
  3492. if (playgroundParticles.applyLifetimeSize && !playgroundParticles.applyParticleArraySize)
  3493. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.lifetimeSize.Evaluate(normalizedLife)*playgroundParticles.scale;
  3494. else if (playgroundParticles.applyLifetimeSize && playgroundParticles.applyParticleArraySize)
  3495. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.lifetimeSize.Evaluate(normalizedLife)*playgroundParticles.particleArraySize.Evaluate((p*1f)/(playgroundParticles.particleCount*1f))*playgroundParticles.scale;
  3496. else if (playgroundParticles.applyParticleArraySize)
  3497. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.particleArraySize.Evaluate((p*1f)/(playgroundParticles.particleCount*1f))*playgroundParticles.scale;
  3498. else playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.scale;
  3499. }
  3500. if (!playgroundParticles.onlySourcePositioning && !playgroundParticles.onlyLifetimePositioning) {
  3501. // Check if particle has sticky collision
  3502. if (playgroundParticles.hasCollisionCache && playgroundParticles.stickyCollisions && playgroundParticles.collisionCache.hasCollided[p] && !playgroundParticles.playgroundCache.isNonBirthed[p]) {
  3503. if (playgroundParticles.stickyCollisionsSurfaceOffset==0)
  3504. playgroundParticles.playgroundCache.position[p] = playgroundParticles.collisionCache.stickyPosition[p];
  3505. else
  3506. playgroundParticles.playgroundCache.position[p] = playgroundParticles.collisionCache.stickyPosition[p] + (playgroundParticles.collisionCache.collisionNormal[p]*playgroundParticles.stickyCollisionsSurfaceOffset);
  3507. } else {
  3508. if (!playgroundParticles.playgroundCache.noForce[p] && playgroundParticles.playgroundCache.life[p]>0) {
  3509. // Velocity bending
  3510. if (playgroundParticles.applyVelocityBending) {
  3511. if (playgroundParticles.velocityBendingType==VELOCITYBENDINGTYPEC.SourcePosition) {
  3512. playgroundParticles.playgroundCache.velocity[p] += Vector3.Reflect(
  3513. new Vector3(
  3514. playgroundParticles.playgroundCache.velocity[p].x*playgroundParticles.velocityBending.x,
  3515. playgroundParticles.playgroundCache.velocity[p].y*playgroundParticles.velocityBending.y,
  3516. playgroundParticles.playgroundCache.velocity[p].z*playgroundParticles.velocityBending.z
  3517. ),
  3518. (playgroundParticles.playgroundCache.targetPosition[p]-playgroundParticles.playgroundCache.position[p]).normalized
  3519. )*t;
  3520. } else {
  3521. playgroundParticles.playgroundCache.velocity[p] += Vector3.Reflect(
  3522. new Vector3(
  3523. playgroundParticles.playgroundCache.velocity[p].x*playgroundParticles.velocityBending.x,
  3524. playgroundParticles.playgroundCache.velocity[p].y*playgroundParticles.velocityBending.y,
  3525. playgroundParticles.playgroundCache.velocity[p].z*playgroundParticles.velocityBending.z
  3526. ),
  3527. (playgroundParticles.playgroundCache.previousParticlePosition[p]-playgroundParticles.playgroundCache.position[p]).normalized
  3528. )*t;
  3529. }
  3530. }
  3531. // Set previous target position (used by delta velocity & local space movement compensation)
  3532. playgroundParticles.playgroundCache.previousTargetPosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  3533. // Gravity
  3534. if (playgroundParticles.localSpace && playgroundParticles.source==SOURCEC.Transform && playgroundParticles.transformIndex!=playgroundParticles.psTransformNum)
  3535. playgroundParticles.playgroundCache.velocity[p] -= playgroundParticles.sourceTransforms[playgroundParticles.transformIndex].rotation*playgroundParticles.gravity*t;
  3536. else
  3537. playgroundParticles.playgroundCache.velocity[p] -= playgroundParticles.gravity*t;
  3538. // Lifetime additive velocity
  3539. if (playgroundParticles.applyLifetimeVelocity)
  3540. playgroundParticles.playgroundCache.velocity[p] += playgroundParticles.lifetimeVelocity.Evaluate(normalizedLife, playgroundParticles.lifetimeVelocityScale)*t;
  3541. // Turbulence inside the calculation loop
  3542. if (playgroundParticles.HasTurbulence() && PlaygroundC.reference.turbulenceThreadMethod==ThreadMethodComponent.InsideParticleCalculation) {
  3543. if (!playgroundParticles.playgroundCache.noForce[p])
  3544. Turbulence(playgroundParticles, playgroundParticles.turbulenceSimplex, p, playgroundParticles.t, playgroundParticles.turbulenceType, playgroundParticles.turbulenceTimeScale, playgroundParticles.turbulenceScale/playgroundParticles.velocityScale, playgroundParticles.turbulenceStrength, playgroundParticles.turbulenceApplyLifetimeStrength, playgroundParticles.turbulenceLifetimeStrength);
  3545. }
  3546. // Damping, max velocity, constraints and final positioning
  3547. // Max velocity
  3548. if (playgroundParticles.playgroundCache.velocity[p].sqrMagnitude>playgroundParticles.maxVelocity)
  3549. playgroundParticles.playgroundCache.velocity[p] = Vector3.ClampMagnitude(playgroundParticles.playgroundCache.velocity[p], playgroundParticles.maxVelocity);
  3550. // Damping
  3551. if (playgroundParticles.damping>0)
  3552. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], zero, playgroundParticles.damping*t);
  3553. // Transition back to source
  3554. if (playgroundParticles.transitionBackToSource) {
  3555. float transitionAmount = playgroundParticles.transitionBackToSourceAmount.Evaluate(normalizedLife)*playgroundParticles.particleTimescale;
  3556. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(playgroundParticles.playgroundCache.position[p], playgroundParticles.playgroundCache.targetPosition[p], transitionAmount);
  3557. playgroundParticles.playgroundCache.velocity[p] *= 1f-transitionAmount;
  3558. }
  3559. // Axis constraints
  3560. if (playgroundParticles.axisConstraints.x)
  3561. playgroundParticles.playgroundCache.velocity[p].x = 0;
  3562. if (playgroundParticles.axisConstraints.y)
  3563. playgroundParticles.playgroundCache.velocity[p].y = 0;
  3564. if (playgroundParticles.axisConstraints.z)
  3565. playgroundParticles.playgroundCache.velocity[p].z = 0;
  3566. // Set calculated collision position
  3567. playgroundParticles.playgroundCache.collisionParticlePosition[p] = playgroundParticles.playgroundCache.position[p];
  3568. // Relocate
  3569. playgroundParticles.playgroundCache.position[p] += (playgroundParticles.playgroundCache.velocity[p]*playgroundParticles.velocityScale)*t;
  3570. if (playgroundParticles.localSpace && playgroundParticles.applyLocalSpaceMovementCompensation) {
  3571. if (!playgroundParticles.applyMovementCompensationLifetimeStrength)
  3572. playgroundParticles.playgroundCache.position[p] += playgroundParticles.playgroundCache.localSpaceMovementCompensation[p];
  3573. else
  3574. playgroundParticles.playgroundCache.position[p] += playgroundParticles.playgroundCache.localSpaceMovementCompensation[p]*playgroundParticles.movementCompensationLifetimeStrength.Evaluate(normalizedLife);
  3575. }
  3576. // Set particle velocity to be able to stretch towards movement
  3577. if (playgroundParticles.renderModeStretch && playgroundParticles.realSimulationTime>0) {
  3578. if (playgroundParticles.applyLifetimeStretching) {
  3579. playgroundParticles.particleCache[p].velocity = Vector3.Slerp (playgroundParticles.particleCache[p].velocity, (playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.previousParticlePosition[p])/playgroundParticles.realSimulationTime, t*playgroundParticles.stretchSpeed)*playgroundParticles.stretchLifetime.Evaluate(playgroundParticles.playgroundCache.life[p]/lifeInSeconds);
  3580. } else {
  3581. if (playgroundParticles.stretchSpeed>0)
  3582. playgroundParticles.particleCache[p].velocity = Vector3.Slerp (playgroundParticles.particleCache[p].velocity, (playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.previousParticlePosition[p])/playgroundParticles.realSimulationTime, t*playgroundParticles.stretchSpeed);
  3583. else playgroundParticles.particleCache[p].velocity = playgroundParticles.stretchStartDirection;
  3584. }
  3585. }
  3586. }
  3587. }
  3588. } else {
  3589. // Only Source Positioning / Lifetime Positioning
  3590. if (playgroundParticles.onlyLifetimePositioning && !playgroundParticles.onlySourcePositioning) {
  3591. if (!playgroundParticles.playgroundCache.changedByPropertyTarget[p]) {
  3592. // Lifetime Positioning by Vector3 Animation Curve
  3593. if (playgroundParticles.lifetimePositioningUsesSourceDirection && playgroundParticles.source!=SOURCEC.Script) {
  3594. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.targetPosition[p]+(playgroundParticles.playgroundCache.targetDirection[p]);
  3595. } else {
  3596. if (!playgroundParticles.applyLifetimePositioningPositionScale) {
  3597. playgroundParticles.playgroundCache.position[p] =
  3598. playgroundParticles.playgroundCache.targetPosition[p]+
  3599. playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale);
  3600. } else {
  3601. playgroundParticles.playgroundCache.position[p] =
  3602. playgroundParticles.playgroundCache.targetPosition[p]+
  3603. playgroundParticles.lifetimePositioning.Evaluate(normalizedLife*lifetimePositioningTimeScale, playgroundParticles.lifetimePositioningScale)*
  3604. playgroundParticles.lifetimePositioningPositionScale.Evaluate(normalizedLife*lifetimePositioningTimeScale);
  3605. }
  3606. }
  3607. }
  3608. } else if (playgroundParticles.source!=SOURCEC.Script && !playgroundParticles.playgroundCache.isNonBirthed[p]) {
  3609. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.targetPosition[p];
  3610. }
  3611. // Set particle velocity to be able to stretch towards movement
  3612. if (playgroundParticles.renderModeStretch && playgroundParticles.realSimulationTime>0) {
  3613. if (playgroundParticles.applyLifetimeStretching) {
  3614. playgroundParticles.particleCache[p].velocity = Vector3.Slerp (playgroundParticles.particleCache[p].velocity, (playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.previousParticlePosition[p])/playgroundParticles.realSimulationTime, t*playgroundParticles.stretchSpeed)*playgroundParticles.stretchLifetime.Evaluate(playgroundParticles.playgroundCache.life[p]/lifeInSeconds);
  3615. } else {
  3616. if (playgroundParticles.stretchSpeed>0)
  3617. playgroundParticles.particleCache[p].velocity = Vector3.Slerp (playgroundParticles.particleCache[p].velocity, (playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.previousParticlePosition[p])/playgroundParticles.realSimulationTime, t*playgroundParticles.stretchSpeed);
  3618. else playgroundParticles.particleCache[p].velocity = playgroundParticles.stretchStartDirection;
  3619. }
  3620. }
  3621. playgroundParticles.playgroundCache.previousTargetPosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  3622. }
  3623. // Rotation
  3624. if (t!=0) {
  3625. if (!playgroundParticles.rotateTowardsDirection)
  3626. playgroundParticles.playgroundCache.rotation[p] += playgroundParticles.playgroundCache.rotationSpeed[p]*t;
  3627. else if (playgroundParticles.playgroundCache.life[p]!=0 && playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.previousParticlePosition[p]!=Vector3.zero) {
  3628. playgroundParticles.playgroundCache.rotation[p] = playgroundParticles.playgroundCache.initialRotation[p]+SignedAngle(
  3629. up,
  3630. playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.previousParticlePosition[p],
  3631. playgroundParticles.rotationNormal
  3632. );
  3633. }
  3634. }
  3635. if (!playgroundParticles.syncPositionsOnMainThread && playgroundParticles.playgroundCache.life[p]>0)
  3636. playgroundParticles.particleCache[p].rotation = playgroundParticles.playgroundCache.rotation[p];
  3637. // Set previous particle position
  3638. playgroundParticles.playgroundCache.previousParticlePosition[p] = playgroundParticles.playgroundCache.position[p];
  3639. // Send timed event
  3640. if (playgroundParticles.hasTimerEvent)
  3641. playgroundParticles.SendEvent(EVENTTYPEC.Time, p);
  3642. } else {
  3643. playgroundParticles.particleCache[p].size = 0;
  3644. playgroundParticles.playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  3645. }
  3646. // Calculate lifetime
  3647. float evaluatedLife = (playgroundParticles.localTime-playgroundParticles.playgroundCache.birth[p])/lifeInSeconds;
  3648. // Lifetime
  3649. if (playgroundParticles.playgroundCache.life[p]<playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p]) {
  3650. playgroundParticles.playgroundCache.life[p] = lifeInSeconds*evaluatedLife;
  3651. if (!playgroundParticles.syncPositionsOnMainThread)
  3652. playgroundParticles.particleCache[p].remainingLifetime = Mathf.Clamp (lifeInSeconds*(1f-evaluatedLife), playgroundParticles.minShurikenLifetime, playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p]);
  3653. if (playgroundParticles.lifetimeValueMethod==VALUEMETHOD.RandomBetweenTwoValues && playgroundParticles.playgroundCache.life[p]>lifeInSeconds) {
  3654. // Send death event for particles with lifetime subtraction
  3655. if ((playgroundParticles.hasEvent||playgroundParticles.hasEventManipulatorGlobal||playgroundParticles.hasEventManipulatorLocal) && !playgroundParticles.playgroundCache.isNonBirthed[p])
  3656. SendDeathEvents(playgroundParticles, p);
  3657. playgroundParticles.particleCache[p].size = 0;
  3658. playgroundParticles.particleCache[p].velocity = zero;
  3659. playgroundParticles.playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  3660. }
  3661. } else {
  3662. // Particle exceeded with death property
  3663. if (!playgroundParticles.loop && !playgroundParticles.playgroundCache.isNonBirthed[p]) {
  3664. // Send death event for particles which died of unnatural cause such as property death, the worst type of death
  3665. if (playgroundParticles.hasEvent||playgroundParticles.hasEventManipulatorGlobal||playgroundParticles.hasEventManipulatorLocal)
  3666. SendDeathEvents(playgroundParticles, p);
  3667. playgroundParticles.InactivateParticle(p);
  3668. continue;
  3669. }
  3670. // Loop exceeded normally
  3671. if (!playgroundParticles.loop && playgroundParticles.localTime>playgroundParticles.simulationStarted+(playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p])-.01f) {
  3672. playgroundParticles.loopExceeded = true;
  3673. if (playgroundParticles.loopExceededOnParticle==p && evaluatedLife>2f) {
  3674. if (playgroundParticles.disableOnDone)
  3675. playgroundParticles.queueEmissionHalt = true;
  3676. playgroundParticles.threadHadNoActiveParticles = true;
  3677. playgroundParticles.hasActiveParticles = false;
  3678. }
  3679. if (playgroundParticles.loopExceededOnParticle==-1)
  3680. playgroundParticles.loopExceededOnParticle = p;
  3681. }
  3682. playgroundParticles.particleCache[p].velocity = zero;
  3683. // Send death event for particles with full lifetime length
  3684. if ((playgroundParticles.hasEvent||playgroundParticles.hasEventManipulatorGlobal||playgroundParticles.hasEventManipulatorLocal) && !playgroundParticles.playgroundCache.isNonBirthed[p])
  3685. SendDeathEvents(playgroundParticles, p);
  3686. // New cycle begins
  3687. if (playgroundParticles.localTime>=playgroundParticles.playgroundCache.birth[p]+playgroundParticles.playgroundCache.birthDelay[p] && !playgroundParticles.loopExceeded && playgroundParticles.source!=SOURCEC.Script && playgroundParticles.emit) {
  3688. if (!playgroundParticles.playgroundCache.changedByPropertyDeath[p] || playgroundParticles.playgroundCache.changedByPropertyDeath[p] && playgroundParticles.localTime>playgroundParticles.playgroundCache.death[p]) {
  3689. Rebirth(playgroundParticles, p, playgroundParticles.internalRandom01);
  3690. } else {
  3691. playgroundParticles.particleCache[p].velocity = zero;
  3692. playgroundParticles.particleCache[p].size = 0;
  3693. playgroundParticles.playgroundCache.position[p] = PlaygroundC.initialTargetPosition;
  3694. }
  3695. } else {
  3696. playgroundParticles.InactivateParticle(p);
  3697. continue;
  3698. }
  3699. }
  3700. // Local Manipulators
  3701. for (int m = 0; m<playgroundParticles.manipulators.Count; m++) {
  3702. CalculateManipulator(playgroundParticles, playgroundParticles.manipulators[m], p, t, playgroundParticles.playgroundCache.life[p], playgroundParticles.playgroundCache.position[p], (playgroundParticles.localSpace?playgroundParticles.manipulators[m].transform.localPosition:playgroundParticles.manipulators[m].transform.position)+playgroundParticles.manipulatorFix, playgroundParticles.localSpace);
  3703. }
  3704. // Global Manipulators
  3705. if (playgroundParticles.hasGlobalAffectingManipulators) {
  3706. for (int m = 0; m<PlaygroundC.reference.manipulators.Count; m++) {
  3707. if ((PlaygroundC.reference.manipulators[m].affects.value & 1<<playgroundParticles.GetLayer())!=0)
  3708. CalculateManipulator(playgroundParticles, PlaygroundC.reference.manipulators[m], p, t, playgroundParticles.playgroundCache.life[p], playgroundParticles.playgroundCache.position[p], (playgroundParticles.localSpace?PlaygroundC.reference.manipulators[m].transform.localPosition:PlaygroundC.reference.manipulators[m].transform.position)+playgroundParticles.manipulatorFix, playgroundParticles.localSpace);
  3709. }
  3710. }
  3711. // Set particle size
  3712. if (!playgroundParticles.syncPositionsOnMainThread)
  3713. playgroundParticles.particleCache[p].size = (playgroundParticles.playgroundCache.maskAlpha[p]>0&&playgroundParticles.particleCache[p].position.y!=initYpos)?playgroundParticles.playgroundCache.size[p]:0;
  3714. } else {
  3715. // Particle is set to not rebirth
  3716. playgroundParticles.InactivateParticle(p);
  3717. }
  3718. // Set particle position if no sync
  3719. if (!playgroundParticles.syncPositionsOnMainThread) {
  3720. playgroundParticles.particleCache[p].position = playgroundParticles.playgroundCache.position[p];
  3721. }
  3722. // Particle got calculated
  3723. playgroundParticles.playgroundCache.isCalculatedThisFrame[p] = true;
  3724. }// <- Particle loop ends here
  3725. // Reset for next frame
  3726. if (playgroundParticles.isSettingParticleTime) playgroundParticles.threadHadNoActiveParticles = false;
  3727. else playgroundParticles.threadHadNoActiveParticles = noActiveParticles && !playgroundParticles.emit || noActiveParticles && !playgroundParticles.loop;
  3728. playgroundParticles.cameFromNonEmissionFrame = false;
  3729. playgroundParticles.isDoneThread = true;
  3730. // Turbulence calculated here when using PlaygroundC.turbulenceThreadMethod of ThreadMethodComponent.OnePerSystem
  3731. if (playgroundParticles.HasTurbulence() && PlaygroundC.reference.turbulenceThreadMethod==ThreadMethodComponent.OnePerSystem) {
  3732. PlaygroundC.RunAsync(()=>{
  3733. for (int p = 0; p<playgroundParticles.particleCount; p++) {
  3734. if (!playgroundParticles.playgroundCache.noForce[p])
  3735. Turbulence(playgroundParticles, playgroundParticles.turbulenceSimplex, p, playgroundParticles.t, playgroundParticles.turbulenceType, playgroundParticles.turbulenceTimeScale, playgroundParticles.turbulenceScale/playgroundParticles.velocityScale, playgroundParticles.turbulenceStrength, playgroundParticles.turbulenceApplyLifetimeStrength, playgroundParticles.turbulenceLifetimeStrength);
  3736. }
  3737. });
  3738. }
  3739. }
  3740. /// <summary>
  3741. /// Sends the death events.
  3742. /// </summary>
  3743. /// <param name="playgroundParticles">Playground particles.</param>
  3744. /// <param name="p">Particle index.</param>
  3745. public static void SendDeathEvents (PlaygroundParticlesC playgroundParticles, int p) {
  3746. if ((playgroundParticles.playgroundCache.life[p]>0||playgroundParticles.playgroundCache.changedByPropertyDeath[p]) && !playgroundParticles.playgroundCache.isNonBirthed[p]) {
  3747. if (playgroundParticles.loop || (!playgroundParticles.loop && playgroundParticles.playgroundCache.isFirstLoop[p])) {
  3748. playgroundParticles.SendEvent(EVENTTYPEC.Death, p);
  3749. }
  3750. }
  3751. if (playgroundParticles.hasEventManipulatorLocal) {
  3752. for (int i = 0; i<playgroundParticles.manipulators.Count; i++) {
  3753. if (playgroundParticles.manipulators[i].trackParticles &&
  3754. playgroundParticles.manipulators[i].RemoveParticle (playgroundParticles.particleSystemId, p)) {
  3755. if (playgroundParticles.manipulators[i].sendEventDeath) {
  3756. playgroundParticles.UpdateEventParticle(playgroundParticles.manipulators[i].manipulatorEventParticle, p);
  3757. playgroundParticles.manipulators[i].SendParticleEventDeath();
  3758. }
  3759. }
  3760. }
  3761. }
  3762. if (playgroundParticles.hasEventManipulatorGlobal) {
  3763. for (int i = 0; i<PlaygroundC.reference.manipulators.Count; i++) {
  3764. if (PlaygroundC.reference.manipulators[i].trackParticles &&
  3765. PlaygroundC.reference.manipulators[i].RemoveParticle (playgroundParticles.particleSystemId, p)) {
  3766. if (PlaygroundC.reference.manipulators[i].sendEventDeath) {
  3767. playgroundParticles.UpdateEventParticle(PlaygroundC.reference.manipulators[i].manipulatorEventParticle, p);
  3768. PlaygroundC.reference.manipulators[i].SendParticleEventDeath();
  3769. }
  3770. }
  3771. }
  3772. }
  3773. playgroundParticles.playgroundCache.isNonBirthed[p] = true;
  3774. }
  3775. public static void Turbulence (PlaygroundParticlesC playgroundParticles, SimplexNoise turbulenceSimplex, int p, float t, TURBULENCETYPE turbulenceType, float turbulenceTimeScale, float turbulenceScale, float turbulenceStrength, bool turbulenceApplyLifetimeStrength, AnimationCurve turbulenceLifetimeStrength) {
  3776. float turbulenceStrengthMultiplier = 1f;
  3777. float zeroFixX = 1f;
  3778. float zeroFixY = 2f;
  3779. float zeroFixZ = 3f;
  3780. if (playgroundParticles.playgroundCache.simulate[p] && playgroundParticles.playgroundCache.position[p].y != PlaygroundC.initialTargetPosition.y) {
  3781. if (turbulenceType==TURBULENCETYPE.Simplex) {
  3782. // Simplex Noise
  3783. if (turbulenceTimeScale>0) {
  3784. if (playgroundParticles.turbulenceApplyLifetimeStrength)
  3785. turbulenceStrengthMultiplier = playgroundParticles.turbulenceLifetimeStrength.Evaluate (Mathf.Clamp01(playgroundParticles.playgroundCache.life[p]/((playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p])-playgroundParticles.playgroundCache.lifetimeSubtraction[p])));
  3786. if (turbulenceStrengthMultiplier>0) {
  3787. if (!playgroundParticles.axisConstraints.x)
  3788. playgroundParticles.playgroundCache.velocity[p].x += (float)turbulenceSimplex.noise(
  3789. (playgroundParticles.playgroundCache.position[p].x+zeroFixX)*turbulenceScale,
  3790. (playgroundParticles.playgroundCache.position[p].y+zeroFixY)*turbulenceScale,
  3791. (playgroundParticles.playgroundCache.position[p].z+zeroFixZ)*turbulenceScale,
  3792. playgroundParticles.localTime*turbulenceTimeScale
  3793. )*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3794. if (!playgroundParticles.axisConstraints.y)
  3795. playgroundParticles.playgroundCache.velocity[p].y += (float)turbulenceSimplex.noise(
  3796. (playgroundParticles.playgroundCache.position[p].y+zeroFixY)*turbulenceScale,
  3797. (playgroundParticles.playgroundCache.position[p].x+zeroFixX)*turbulenceScale,
  3798. (playgroundParticles.playgroundCache.position[p].z+zeroFixZ)*turbulenceScale,
  3799. playgroundParticles.localTime*turbulenceTimeScale
  3800. )*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3801. if (!playgroundParticles.axisConstraints.z)
  3802. playgroundParticles.playgroundCache.velocity[p].z += (float)turbulenceSimplex.noise(
  3803. (playgroundParticles.playgroundCache.position[p].z+zeroFixZ)*turbulenceScale,
  3804. (playgroundParticles.playgroundCache.position[p].x+zeroFixX)*turbulenceScale,
  3805. (playgroundParticles.playgroundCache.position[p].y+zeroFixY)*turbulenceScale,
  3806. playgroundParticles.localTime*turbulenceTimeScale
  3807. )*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3808. }
  3809. } else {
  3810. if (turbulenceStrengthMultiplier>0) {
  3811. if (!playgroundParticles.axisConstraints.x)
  3812. playgroundParticles.playgroundCache.velocity[p].x += (float)turbulenceSimplex.noise(
  3813. (playgroundParticles.playgroundCache.position[p].x+zeroFixX)*turbulenceScale,
  3814. (playgroundParticles.playgroundCache.position[p].y+zeroFixY)*turbulenceScale,
  3815. (playgroundParticles.playgroundCache.position[p].z+zeroFixZ)*turbulenceScale
  3816. )*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3817. if (!playgroundParticles.axisConstraints.y)
  3818. playgroundParticles.playgroundCache.velocity[p].y += (float)turbulenceSimplex.noise(
  3819. (playgroundParticles.playgroundCache.position[p].y+zeroFixY)*turbulenceScale,
  3820. (playgroundParticles.playgroundCache.position[p].x+zeroFixX)*turbulenceScale,
  3821. (playgroundParticles.playgroundCache.position[p].z+zeroFixZ)*turbulenceScale
  3822. )*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3823. if (!playgroundParticles.axisConstraints.z)
  3824. playgroundParticles.playgroundCache.velocity[p].z += (float)turbulenceSimplex.noise(
  3825. (playgroundParticles.playgroundCache.position[p].z+zeroFixZ)*turbulenceScale,
  3826. (playgroundParticles.playgroundCache.position[p].x+zeroFixX)*turbulenceScale,
  3827. (playgroundParticles.playgroundCache.position[p].y+zeroFixY)*turbulenceScale
  3828. )*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3829. }
  3830. }
  3831. } else {
  3832. // Perlin Noise
  3833. if (playgroundParticles.turbulenceStrength>0) {
  3834. if (playgroundParticles.turbulenceApplyLifetimeStrength)
  3835. turbulenceStrengthMultiplier = playgroundParticles.turbulenceLifetimeStrength.Evaluate (Mathf.Clamp01(playgroundParticles.playgroundCache.life[p]/((playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p])-playgroundParticles.playgroundCache.lifetimeSubtraction[p])));
  3836. if (turbulenceStrengthMultiplier>0) {
  3837. if (!playgroundParticles.axisConstraints.x)
  3838. playgroundParticles.playgroundCache.velocity[p].x += (Mathf.PerlinNoise (
  3839. playgroundParticles.localTime*turbulenceTimeScale,
  3840. playgroundParticles.playgroundCache.position[p].z*turbulenceScale
  3841. )-.5f)*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3842. if (!playgroundParticles.axisConstraints.y)
  3843. playgroundParticles.playgroundCache.velocity[p].y += (Mathf.PerlinNoise (
  3844. playgroundParticles.localTime*turbulenceTimeScale,
  3845. playgroundParticles.playgroundCache.position[p].x*turbulenceScale
  3846. )-.5f)*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3847. if (!playgroundParticles.axisConstraints.z)
  3848. playgroundParticles.playgroundCache.velocity[p].z += (Mathf.PerlinNoise (
  3849. playgroundParticles.localTime*turbulenceTimeScale,
  3850. playgroundParticles.playgroundCache.position[p].y*turbulenceScale
  3851. )-.5f)*turbulenceStrength*t*turbulenceStrengthMultiplier;
  3852. }
  3853. }
  3854. }
  3855. }
  3856. }
  3857. public void CalculateCollisions () {
  3858. Collisions (thisInstance);
  3859. }
  3860. /// <summary>
  3861. /// Calculates particle collisions, this runs automatically if collision is set to true. (Must currently run on main-thread due to the Physics.Raycast dependency.)
  3862. /// </summary>
  3863. /// <param name="playgroundParticles">Particle Playground system.</param>
  3864. public static void Collisions (PlaygroundParticlesC playgroundParticles) {
  3865. if (!playgroundParticles.onlySourcePositioning && !playgroundParticles.onlyLifetimePositioning && playgroundParticles.collisionRadius>0 && playgroundParticles.hasEmitted) {
  3866. Ray ray = new Ray();
  3867. float distance;
  3868. bool is3d = playgroundParticles.collisionType==COLLISIONTYPEC.Physics3D;
  3869. bool hasCollisionExclusion = playgroundParticles.collisionExclusion!=null && playgroundParticles.collisionExclusion.Count>0;
  3870. bool hasRandomBounce = playgroundParticles.bounceRandomMin!=Vector3.zero||playgroundParticles.bounceRandomMax!=Vector3.zero;
  3871. RaycastHit hitInfo;
  3872. RaycastHit2D hitInfo2D;
  3873. bool hasEvents = playgroundParticles.events.Count>0;
  3874. // Prepare the collision cache
  3875. if ((playgroundParticles.stickyCollisions || playgroundParticles.forceCollisionCaching) && playgroundParticles.collisionCache==null)
  3876. playgroundParticles.collisionCache = new CollisionCache(playgroundParticles.particleCount);
  3877. // Prepare the infinite collision planes
  3878. if (playgroundParticles.collision && playgroundParticles.colliders.Count>0) {
  3879. for (int c = 0; c<playgroundParticles.colliders.Count; c++) {
  3880. playgroundParticles.colliders[c].UpdatePlane();
  3881. }
  3882. }
  3883. // Check cache length
  3884. if (playgroundParticles.playgroundCache.noForce.Length!=playgroundParticles.particleCount)
  3885. playgroundParticles.playgroundCache.noForce = new bool[playgroundParticles.particleCount];
  3886. if (playgroundParticles.playgroundCache.lifetimeSubtraction.Length!=playgroundParticles.particleCount)
  3887. playgroundParticles.playgroundCache.lifetimeSubtraction = new float[playgroundParticles.particleCount];
  3888. for (int p = 0; p<playgroundParticles.particleCount; p++) {
  3889. // Early out if there's no need to continue with this particle
  3890. if (playgroundParticles.playgroundCache.life[p]==0 || playgroundParticles.playgroundCache.life[p]>=(playgroundParticles.playgroundCache.death[p]-playgroundParticles.playgroundCache.birth[p])-playgroundParticles.playgroundCache.lifetimeSubtraction[p] || playgroundParticles.playgroundCache.noForce[p]) continue;
  3891. // Sticky? Do a sticky update then continue
  3892. if (playgroundParticles.stickyCollisions && playgroundParticles.collisionCache.hasCollided[p]) {
  3893. playgroundParticles.collisionCache.UpdateStickyPosition(p);
  3894. playgroundParticles.playgroundCache.position[p] = playgroundParticles.collisionCache.stickyPosition[p];
  3895. continue;
  3896. }
  3897. // Playground Plane colliders (never exceed these)
  3898. for (int c = 0; c<playgroundParticles.colliders.Count; c++) {
  3899. if (playgroundParticles.colliders[c].enabled && playgroundParticles.colliders[c].transform && !playgroundParticles.colliders[c].plane.GetSide(playgroundParticles.playgroundCache.position[p])) {
  3900. // Set particle to location
  3901. ray.origin = playgroundParticles.playgroundCache.position[p];
  3902. ray.direction = playgroundParticles.colliders[c].plane.normal;
  3903. if (playgroundParticles.colliders[c].plane.Raycast(ray, out distance))
  3904. playgroundParticles.playgroundCache.position[p] = ray.GetPoint(distance);
  3905. // Store velocity before collision
  3906. Vector3 preCollisionVelocity = playgroundParticles.playgroundCache.velocity[p];
  3907. // Update collision cache
  3908. if (!playgroundParticles.stickyCollisions) {
  3909. if (playgroundParticles.forceCollisionCaching)
  3910. playgroundParticles.collisionCache.Set (p, playgroundParticles.playgroundCache.position[p], playgroundParticles.colliders[c].plane.normal, playgroundParticles.colliders[c].transform);
  3911. } else
  3912. playgroundParticles.collisionCache.SetSticky (p, playgroundParticles.playgroundCache.position[p], playgroundParticles.colliders[c].plane.normal, playgroundParticles.stickyCollisionsSurfaceOffset, playgroundParticles.colliders[c].transform);
  3913. // Reflect particle
  3914. playgroundParticles.playgroundCache.velocity[p] = Vector3.Reflect(playgroundParticles.playgroundCache.velocity[p], hasRandomBounce?playgroundParticles.colliders[c].plane.normal+RandomVector3(playgroundParticles.internalRandom01, playgroundParticles.bounceRandomMin, playgroundParticles.bounceRandomMax):playgroundParticles.colliders[c].plane.normal)*playgroundParticles.bounciness;
  3915. // Apply lifetime loss
  3916. if (playgroundParticles.lifetimeLoss>0) {
  3917. playgroundParticles.playgroundCache.birth[p] -= playgroundParticles.playgroundCache.life[p]/(1f-playgroundParticles.lifetimeLoss);
  3918. playgroundParticles.playgroundCache.changedByPropertyDeath[p] = true;
  3919. }
  3920. // Send event
  3921. if (hasEvents)
  3922. playgroundParticles.SendEvent(EVENTTYPEC.Collision, p, preCollisionVelocity, playgroundParticles.colliders[c].transform);
  3923. }
  3924. }
  3925. // Colliders in scene
  3926. if (playgroundParticles.playgroundCache.velocity[p].magnitude>PlaygroundC.collisionSleepVelocity) {
  3927. // Collide by checking for potential passed collider in the direction of this particle's velocity from the previous frame
  3928. if (is3d) {
  3929. if (Physics.Raycast(
  3930. playgroundParticles.playgroundCache.collisionParticlePosition[p],
  3931. (playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.collisionParticlePosition[p]),
  3932. out hitInfo,
  3933. playgroundParticles.collisionPrecision? (Vector3.Distance (playgroundParticles.playgroundCache.position[p], playgroundParticles.playgroundCache.collisionParticlePosition[p])+playgroundParticles.collisionRadius)*2f : (Vector3.SqrMagnitude(playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.collisionParticlePosition[p])+playgroundParticles.collisionRadius)*2f,
  3934. playgroundParticles.collisionMask))
  3935. {
  3936. // Check that this object isn't excluded
  3937. if (hasCollisionExclusion) {
  3938. if (playgroundParticles.collisionExclusion.Contains(hitInfo.transform))
  3939. continue;
  3940. }
  3941. // Set particle to location
  3942. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.collisionParticlePosition[p];
  3943. // Update collision cache
  3944. if (!playgroundParticles.stickyCollisions) {
  3945. if (playgroundParticles.forceCollisionCaching)
  3946. playgroundParticles.collisionCache.Set (p, hitInfo.point, hitInfo.normal, hitInfo.transform);
  3947. } else if ((playgroundParticles.stickyCollisionsMask.value & 1<<hitInfo.transform.gameObject.layer)!=0)
  3948. playgroundParticles.collisionCache.SetSticky (p, hitInfo.point, hitInfo.normal, playgroundParticles.stickyCollisionsSurfaceOffset, hitInfo.transform);
  3949. // Store velocity before collision
  3950. Vector3 preCollisionVelocity = playgroundParticles.playgroundCache.velocity[p];
  3951. // Reflect particle
  3952. playgroundParticles.playgroundCache.velocity[p] = Vector3.Reflect(playgroundParticles.playgroundCache.velocity[p], hasRandomBounce?hitInfo.normal+RandomVector3(playgroundParticles.internalRandom01, playgroundParticles.bounceRandomMin, playgroundParticles.bounceRandomMax) : hitInfo.normal)*playgroundParticles.bounciness;
  3953. // Apply lifetime loss
  3954. if (playgroundParticles.lifetimeLoss>0) {
  3955. playgroundParticles.playgroundCache.birth[p] -= playgroundParticles.playgroundCache.life[p]/(1f-playgroundParticles.lifetimeLoss);
  3956. playgroundParticles.playgroundCache.changedByPropertyDeath[p] = true;
  3957. }
  3958. // Add force to rigidbody
  3959. if (playgroundParticles.affectRigidbodies && hitInfo.rigidbody)
  3960. hitInfo.rigidbody.AddForceAtPosition((playgroundParticles.inverseRigidbodyCollision?-preCollisionVelocity:preCollisionVelocity)*playgroundParticles.mass, playgroundParticles.playgroundCache.position[p]);
  3961. // Send event
  3962. if (hasEvents)
  3963. playgroundParticles.SendEvent(EVENTTYPEC.Collision, p, preCollisionVelocity, hitInfo.transform, hitInfo.collider);
  3964. if (playgroundParticles.hasEventManipulatorLocal) {
  3965. for (int i = 0; i<playgroundParticles.manipulators.Count; i++) {
  3966. if (playgroundParticles.manipulators[i].trackParticles &&
  3967. playgroundParticles.manipulators[i].sendEventCollision &&
  3968. playgroundParticles.manipulators[i].Contains (playgroundParticles.playgroundCache.position[p], playgroundParticles.manipulators[i].transform.position)) {
  3969. playgroundParticles.UpdateEventParticle(playgroundParticles.manipulators[i].manipulatorEventParticle, p);
  3970. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionCollider = hitInfo.collider;
  3971. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionParticlePosition = hitInfo.point;
  3972. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionTransform = hitInfo.transform;
  3973. playgroundParticles.manipulators[i].SendParticleEventCollision();
  3974. }
  3975. }
  3976. }
  3977. if (playgroundParticles.hasEventManipulatorGlobal) {
  3978. for (int i = 0; i<PlaygroundC.reference.manipulators.Count; i++) {
  3979. if (PlaygroundC.reference.manipulators[i].trackParticles &&
  3980. PlaygroundC.reference.manipulators[i].sendEventCollision &&
  3981. PlaygroundC.reference.manipulators[i].Contains (playgroundParticles.playgroundCache.position[p], PlaygroundC.reference.manipulators[i].transform.position)) {
  3982. playgroundParticles.UpdateEventParticle(PlaygroundC.reference.manipulators[i].manipulatorEventParticle, p);
  3983. PlaygroundC.reference.manipulators[i].manipulatorEventParticle.collisionCollider = hitInfo.collider;
  3984. PlaygroundC.reference.manipulators[i].manipulatorEventParticle.collisionParticlePosition = hitInfo.point;
  3985. PlaygroundC.reference.manipulators[i].manipulatorEventParticle.collisionTransform = hitInfo.transform;
  3986. PlaygroundC.reference.manipulators[i].SendParticleEventCollision();
  3987. }
  3988. }
  3989. }
  3990. }
  3991. } else {
  3992. hitInfo2D = Physics2D.Raycast(
  3993. playgroundParticles.playgroundCache.collisionParticlePosition[p],
  3994. (playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.collisionParticlePosition[p]),
  3995. playgroundParticles.collisionPrecision? (Vector3.Distance (playgroundParticles.playgroundCache.position[p], playgroundParticles.playgroundCache.collisionParticlePosition[p])+playgroundParticles.collisionRadius)*2f : (Vector3.SqrMagnitude(playgroundParticles.playgroundCache.position[p]-playgroundParticles.playgroundCache.collisionParticlePosition[p])+playgroundParticles.collisionRadius)*2f,
  3996. playgroundParticles.collisionMask,
  3997. playgroundParticles.minCollisionDepth,
  3998. playgroundParticles.maxCollisionDepth
  3999. );
  4000. if (hitInfo2D.collider!=null) {
  4001. // Check that this object isn't excluded
  4002. if (hasCollisionExclusion) {
  4003. if (playgroundParticles.collisionExclusion.Contains(hitInfo2D.transform))
  4004. continue;
  4005. }
  4006. // Set particle to location
  4007. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.collisionParticlePosition[p];
  4008. // Update collision cache
  4009. if (!playgroundParticles.stickyCollisions) {
  4010. if (playgroundParticles.forceCollisionCaching)
  4011. playgroundParticles.collisionCache.Set (p, hitInfo2D.point, hitInfo2D.normal, hitInfo2D.transform);
  4012. } else if ((playgroundParticles.stickyCollisionsMask.value & 1<<hitInfo2D.transform.gameObject.layer)!=0)
  4013. playgroundParticles.collisionCache.SetSticky (p, hitInfo2D.point, hitInfo2D.normal, playgroundParticles.stickyCollisionsSurfaceOffset, hitInfo2D.transform);
  4014. // Store velocity before collision
  4015. Vector3 preCollisionVelocity = playgroundParticles.playgroundCache.velocity[p];
  4016. // Reflect particle
  4017. playgroundParticles.playgroundCache.velocity[p] = Vector3.Reflect(playgroundParticles.playgroundCache.velocity[p], hasRandomBounce?(Vector3)hitInfo2D.normal+RandomVector3(playgroundParticles.internalRandom01, playgroundParticles.bounceRandomMin, playgroundParticles.bounceRandomMax) : (Vector3)hitInfo2D.normal)*playgroundParticles.bounciness;
  4018. // Apply lifetime loss
  4019. if (playgroundParticles.lifetimeLoss>0) {
  4020. playgroundParticles.playgroundCache.birth[p] -= playgroundParticles.playgroundCache.life[p]/(1f-playgroundParticles.lifetimeLoss);
  4021. playgroundParticles.playgroundCache.changedByPropertyDeath[p] = true;
  4022. }
  4023. // Add force to rigidbody
  4024. if (playgroundParticles.affectRigidbodies && hitInfo2D.rigidbody)
  4025. hitInfo2D.rigidbody.AddForceAtPosition((playgroundParticles.inverseRigidbodyCollision?-preCollisionVelocity:preCollisionVelocity)*playgroundParticles.mass, playgroundParticles.playgroundCache.position[p]);
  4026. // Send event
  4027. if (hasEvents)
  4028. playgroundParticles.SendEvent(EVENTTYPEC.Collision, p, preCollisionVelocity, hitInfo2D.transform, hitInfo2D.collider);
  4029. if (playgroundParticles.hasEventManipulatorLocal) {
  4030. for (int i = 0; i<playgroundParticles.manipulators.Count; i++) {
  4031. if (playgroundParticles.manipulators[i].trackParticles &&
  4032. playgroundParticles.manipulators[i].sendEventCollision &&
  4033. playgroundParticles.manipulators[i].Contains (playgroundParticles.playgroundCache.position[p], playgroundParticles.manipulators[i].transform.position)) {
  4034. playgroundParticles.UpdateEventParticle(playgroundParticles.manipulators[i].manipulatorEventParticle, p);
  4035. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionCollider2D = hitInfo2D.collider;
  4036. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionParticlePosition = hitInfo2D.point;
  4037. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionTransform = hitInfo2D.transform;
  4038. playgroundParticles.manipulators[i].SendParticleEventCollision();
  4039. }
  4040. }
  4041. }
  4042. if (playgroundParticles.hasEventManipulatorGlobal) {
  4043. for (int i = 0; i<PlaygroundC.reference.manipulators.Count; i++) {
  4044. if (PlaygroundC.reference.manipulators[i].trackParticles &&
  4045. PlaygroundC.reference.manipulators[i].sendEventCollision &&
  4046. PlaygroundC.reference.manipulators[i].Contains (playgroundParticles.playgroundCache.position[p], PlaygroundC.reference.manipulators[i].transform.position)) {
  4047. playgroundParticles.UpdateEventParticle(PlaygroundC.reference.manipulators[i].manipulatorEventParticle, p);
  4048. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionCollider2D = hitInfo2D.collider;
  4049. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionParticlePosition = hitInfo2D.point;
  4050. playgroundParticles.manipulators[i].manipulatorEventParticle.collisionTransform = hitInfo2D.transform;
  4051. PlaygroundC.reference.manipulators[i].SendParticleEventCollision();
  4052. }
  4053. }
  4054. }
  4055. }
  4056. }
  4057. } else {
  4058. playgroundParticles.playgroundCache.velocity[p] = Vector3.zero;
  4059. }
  4060. }
  4061. }
  4062. }
  4063. /// <summary>
  4064. /// Returns the offset as a remainder from a point, constructed for multithreading.
  4065. /// </summary>
  4066. /// <returns>The overflow position.</returns>
  4067. /// <param name="overflow">Overflow by Vector3.</param>
  4068. /// <param name="currentVal">Current value.</param>
  4069. /// <param name="maxVal">Max value.</param>
  4070. public static Vector3 GetOverflow2 (Vector3 overflow, int currentVal, int maxVal) {
  4071. float iteration = (currentVal/maxVal);
  4072. return new Vector3(
  4073. overflow.x*iteration,
  4074. overflow.y*iteration,
  4075. overflow.z*iteration
  4076. );
  4077. }
  4078. /// <summary>
  4079. /// Returns the offset with direction as a remainder from a point, constructed for multithreading.
  4080. /// </summary>
  4081. /// <returns>The overflow position in direction.</returns>
  4082. /// <param name="overflow">Overflow by Vector3.</param>
  4083. /// <param name="direction">Direction.</param>
  4084. /// <param name="currentVal">Current value.</param>
  4085. /// <param name="maxVal">Max value.</param>
  4086. public static Vector3 GetOverflow2 (Vector3 overflow, Vector3 direction, int currentVal, int maxVal) {
  4087. float iteration = (currentVal/maxVal);
  4088. return new Vector3(
  4089. direction.x*overflow.x*iteration,
  4090. direction.y*overflow.y*iteration,
  4091. direction.z*overflow.z*iteration
  4092. );
  4093. }
  4094. /// <summary>
  4095. /// Multiplies two Vector3.
  4096. /// </summary>
  4097. /// <returns>The scale of two Vector3.</returns>
  4098. /// <param name="v1">First Vector3.</param>
  4099. /// <param name="v2">Second Vector3.</param>
  4100. public static Vector3 Vector3Scale (Vector3 v1, Vector3 v2) {
  4101. return new Vector3(v1.x*v2.x,v1.y*v2.y,v1.z*v2.z);
  4102. }
  4103. /// <summary>
  4104. /// Calculates the effect from a Manipulator.
  4105. /// </summary>
  4106. /// <param name="playgroundParticles">Playground particles.</param>
  4107. /// <param name="thisManipulator">This manipulator.</param>
  4108. /// <param name="p">Particle index.</param>
  4109. /// <param name="t">Delta time.</param>
  4110. /// <param name="life">Lifetime.</param>
  4111. /// <param name="particlePosition">Particle position.</param>
  4112. /// <param name="manipulatorPosition">Manipulator position.</param>
  4113. /// <param name="manipulatorDistance">Manipulator distance.</param>
  4114. /// <param name="localSpace">Is calculation in local space?</param>
  4115. public static void CalculateManipulator (PlaygroundParticlesC playgroundParticles, ManipulatorObjectC thisManipulator, int p, float t, float life, Vector3 particlePosition, Vector3 manipulatorPosition, bool localSpace) {
  4116. if (thisManipulator.enabled && thisManipulator.transform.available && thisManipulator.strength!=0 && thisManipulator.LifetimeFilter(life, playgroundParticles.lifetime) && thisManipulator.ParticleFilter (p, playgroundParticles.particleCount)) {
  4117. bool contains = thisManipulator.Contains(localSpace?(playgroundParticles.particleSystemRotation*particlePosition):particlePosition, localSpace?(playgroundParticles.particleSystemRotation*manipulatorPosition):manipulatorPosition);
  4118. // Is this a particle which shouldn't be affected by this manipulator?
  4119. if (contains && (playgroundParticles.playgroundCache.excludeFromManipulatorId[p]==thisManipulator.manipulatorId || thisManipulator.nonAffectedParticles.Count>0 && thisManipulator.ContainsNonAffectedParticle(playgroundParticles.particleSystemId, p))) {
  4120. return;
  4121. }
  4122. // Manipulator events
  4123. if (thisManipulator.trackParticles) {
  4124. // Particle entering
  4125. if (contains) {
  4126. if ((thisManipulator.trackingMethod==TrackingMethod.ManipulatorId && !thisManipulator.IsSameId(playgroundParticles.playgroundCache.manipulatorId[p])) || (thisManipulator.trackingMethod==TrackingMethod.ParticleId && !thisManipulator.ContainsParticle(playgroundParticles.particleSystemId, p))) {
  4127. playgroundParticles.playgroundCache.manipulatorId[p] = thisManipulator.manipulatorId;
  4128. thisManipulator.AddParticle(playgroundParticles.particleSystemId, p);
  4129. if (thisManipulator.sendEventEnter) {
  4130. playgroundParticles.UpdateEventParticle(thisManipulator.manipulatorEventParticle, p);
  4131. thisManipulator.SendParticleEventEnter();
  4132. }
  4133. }
  4134. } else {
  4135. // Particle exiting
  4136. if ((thisManipulator.trackingMethod==TrackingMethod.ManipulatorId && thisManipulator.IsSameId(playgroundParticles.playgroundCache.manipulatorId[p])) || (thisManipulator.trackingMethod==TrackingMethod.ParticleId && thisManipulator.ContainsParticle(playgroundParticles.particleSystemId, p))) {
  4137. playgroundParticles.playgroundCache.manipulatorId[p] = 0;
  4138. thisManipulator.RemoveParticle(playgroundParticles.particleSystemId, p);
  4139. if (thisManipulator.sendEventExit) {
  4140. playgroundParticles.UpdateEventParticle(thisManipulator.manipulatorEventParticle, p);
  4141. thisManipulator.SendParticleEventExit();
  4142. }
  4143. }
  4144. }
  4145. }
  4146. float manipulatorDistance = 0;
  4147. bool hasConstraints = thisManipulator.axisConstraints.HasConstraints();
  4148. Vector3 constrainedVelocity = playgroundParticles.playgroundCache.velocity[p];
  4149. if (!playgroundParticles.onlySourcePositioning && !playgroundParticles.playgroundCache.noForce[p]) {
  4150. // Attractors
  4151. if (thisManipulator.type==MANIPULATORTYPEC.Attractor) {
  4152. if (contains) {
  4153. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4154. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], (manipulatorPosition-particlePosition)*(thisManipulator.strength/manipulatorDistance), t*(thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4155. }
  4156. } else
  4157. // Attractors Gravitational
  4158. if (thisManipulator.type==MANIPULATORTYPEC.AttractorGravitational) {
  4159. if (contains) {
  4160. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4161. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], (manipulatorPosition-particlePosition)*thisManipulator.strength/manipulatorDistance, t/thisManipulator.strengthSmoothing);
  4162. }
  4163. } else
  4164. // Repellents
  4165. if (thisManipulator.type==MANIPULATORTYPEC.Repellent) {
  4166. if (contains) {
  4167. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4168. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], (particlePosition-manipulatorPosition)*(thisManipulator.strength/manipulatorDistance), t*(thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4169. }
  4170. } else
  4171. // Vortex
  4172. if (thisManipulator.type==MANIPULATORTYPEC.Vortex) {
  4173. if (contains) {
  4174. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4175. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], ((manipulatorPosition-particlePosition)*thisManipulator.strength/manipulatorDistance)-Vector3.Cross(thisManipulator.transform.up, (manipulatorPosition-particlePosition))*thisManipulator.strength/manipulatorDistance, (t*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4176. }
  4177. }
  4178. }
  4179. // Properties
  4180. if (thisManipulator.type==MANIPULATORTYPEC.Property) {
  4181. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4182. PropertyManipulator(playgroundParticles, thisManipulator, thisManipulator.property, p, t, particlePosition, manipulatorPosition, manipulatorDistance, localSpace, contains);
  4183. } else
  4184. // Combined
  4185. if (thisManipulator.type==MANIPULATORTYPEC.Combined) {
  4186. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4187. for (int i = 0; i<thisManipulator.properties.Count; i++)
  4188. PropertyManipulator(playgroundParticles, thisManipulator, thisManipulator.properties[i], p, t, particlePosition, manipulatorPosition, manipulatorDistance, localSpace, contains);
  4189. }
  4190. // Apply constraints
  4191. if (hasConstraints) {
  4192. playgroundParticles.playgroundCache.velocity[p] = new Vector3(
  4193. thisManipulator.axisConstraints.x? constrainedVelocity.x : playgroundParticles.playgroundCache.velocity[p].x,
  4194. thisManipulator.axisConstraints.y? constrainedVelocity.y : playgroundParticles.playgroundCache.velocity[p].y,
  4195. thisManipulator.axisConstraints.z? constrainedVelocity.z : playgroundParticles.playgroundCache.velocity[p].z
  4196. );
  4197. playgroundParticles.playgroundCache.position[p] = new Vector3(
  4198. thisManipulator.axisConstraints.x? particlePosition.x : playgroundParticles.playgroundCache.position[p].x,
  4199. thisManipulator.axisConstraints.y? particlePosition.y : playgroundParticles.playgroundCache.position[p].y,
  4200. thisManipulator.axisConstraints.z? particlePosition.z : playgroundParticles.playgroundCache.position[p].z
  4201. );
  4202. }
  4203. }
  4204. }
  4205. /// <summary>
  4206. /// Calculates the effect from Manipulator's properties.
  4207. /// </summary>
  4208. /// <param name="playgroundParticles">Playground particles.</param>
  4209. /// <param name="thisManipulator">This manipulator.</param>
  4210. /// <param name="thisManipulatorProperty">This manipulator property.</param>
  4211. /// <param name="p">Particle index.</param>
  4212. /// <param name="t">Delta time.</param>
  4213. /// <param name="particlePosition">Particle position.</param>
  4214. /// <param name="manipulatorPosition">Manipulator position.</param>
  4215. /// <param name="localSpace">Is calculation in local space?</param>
  4216. /// <param name="contains">Does the Manipulator contain the particle at index?</param>
  4217. public static void PropertyManipulator (PlaygroundParticlesC playgroundParticles, ManipulatorObjectC thisManipulator, ManipulatorPropertyC thisManipulatorProperty, int p, float t, Vector3 particlePosition, Vector3 manipulatorPosition, float manipulatorDistance, bool localSpace, bool contains) {
  4218. if (contains) {
  4219. switch (thisManipulatorProperty.type) {
  4220. // Math Property
  4221. case MANIPULATORPROPERTYTYPEC.Math:
  4222. if (thisManipulatorProperty.mathProperty.type == MATHMANIPULATORTYPE.Sin || thisManipulatorProperty.mathProperty.type == MATHMANIPULATORTYPE.Cos)
  4223. t = playgroundParticles.localTime;
  4224. switch (thisManipulatorProperty.mathProperty.property) {
  4225. case MATHMANIPULATORPROPERTY.Position:
  4226. playgroundParticles.playgroundCache.position[p] = thisManipulatorProperty.mathProperty.EvaluatePosition(playgroundParticles.playgroundCache.position[p], t);
  4227. break;
  4228. case MATHMANIPULATORPROPERTY.Velocity:
  4229. playgroundParticles.playgroundCache.velocity[p] = thisManipulatorProperty.mathProperty.Evaluate(playgroundParticles.playgroundCache.velocity[p], t);
  4230. break;
  4231. case MATHMANIPULATORPROPERTY.Rotation:
  4232. if (!playgroundParticles.rotateTowardsDirection)
  4233. playgroundParticles.playgroundCache.rotation[p] = thisManipulatorProperty.mathProperty.Evaluate(playgroundParticles.playgroundCache.rotation[p], t)%360f;
  4234. break;
  4235. case MATHMANIPULATORPROPERTY.Size:
  4236. playgroundParticles.playgroundCache.size[p] = thisManipulatorProperty.mathProperty.Evaluate(playgroundParticles.playgroundCache.size[p], t);
  4237. playgroundParticles.playgroundCache.changedByPropertySize[p] = true;
  4238. break;
  4239. }
  4240. break;
  4241. // Velocity Property
  4242. case MANIPULATORPROPERTYTYPEC.Velocity:
  4243. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None)
  4244. playgroundParticles.playgroundCache.velocity[p] = thisManipulatorProperty.useLocalRotation?
  4245. thisManipulatorProperty.localVelocity
  4246. :
  4247. thisManipulatorProperty.velocity;
  4248. else {
  4249. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4250. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], thisManipulatorProperty.useLocalRotation?
  4251. thisManipulatorProperty.localVelocity
  4252. :
  4253. thisManipulatorProperty.velocity, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4254. }
  4255. break;
  4256. // Additive Velocity Property
  4257. case MANIPULATORPROPERTYTYPEC.AdditiveVelocity:
  4258. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4259. playgroundParticles.playgroundCache.velocity[p] += thisManipulatorProperty.useLocalRotation?
  4260. thisManipulatorProperty.localVelocity*((t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing)
  4261. :
  4262. thisManipulatorProperty.velocity*((t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4263. break;
  4264. // Color Property
  4265. case MANIPULATORPROPERTYTYPEC.Color:
  4266. Color staticColor;
  4267. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4268. if (thisManipulatorProperty.keepColorAlphas) {
  4269. staticColor = thisManipulatorProperty.color;
  4270. staticColor.a = Mathf.Clamp(playgroundParticles.lifetimeColor.Evaluate(playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime).a, 0, staticColor.a);
  4271. playgroundParticles.particleCache[p].color = staticColor;
  4272. } else playgroundParticles.particleCache[p].color = thisManipulatorProperty.color;
  4273. } else {
  4274. if (thisManipulatorProperty.keepColorAlphas) {
  4275. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4276. staticColor = thisManipulatorProperty.color;
  4277. staticColor.a = Mathf.Clamp(playgroundParticles.lifetimeColor.Evaluate(playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime).a, 0, staticColor.a);
  4278. playgroundParticles.particleCache[p].color = Color.Lerp(playgroundParticles.particleCache[p].color, staticColor, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4279. } else playgroundParticles.particleCache[p].color = Color.Lerp(playgroundParticles.particleCache[p].color, thisManipulatorProperty.color, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4280. playgroundParticles.playgroundCache.changedByPropertyColorLerp[p] = true;
  4281. }
  4282. // Only color in range of manipulator boundaries
  4283. playgroundParticles.playgroundCache.changedByPropertyColor[p] = true;
  4284. // Keep alpha of original color
  4285. if (thisManipulatorProperty.keepColorAlphas)
  4286. playgroundParticles.playgroundCache.changedByPropertyColorKeepAlpha[p] = true;
  4287. // Set color pairing key
  4288. if (playgroundParticles.playgroundCache.propertyColorId[p] != thisManipulator.manipulatorId) {
  4289. playgroundParticles.playgroundCache.propertyColorId[p] = thisManipulator.manipulatorId;
  4290. }
  4291. break;
  4292. // Lifetime Color Property
  4293. case MANIPULATORPROPERTYTYPEC.LifetimeColor:
  4294. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4295. playgroundParticles.particleCache[p].color = thisManipulatorProperty.lifetimeColor.Evaluate(playgroundParticles.lifetime>0?playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime:0);
  4296. } else {
  4297. playgroundParticles.particleCache[p].color = Color.Lerp(playgroundParticles.particleCache[p].color, thisManipulatorProperty.lifetimeColor.Evaluate(playgroundParticles.lifetime>0?playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime:0), (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4298. playgroundParticles.playgroundCache.changedByPropertyColorLerp[p] = true;
  4299. }
  4300. // Only color in range of manipulator boundaries
  4301. playgroundParticles.playgroundCache.changedByPropertyColor[p] = true;
  4302. // Set color pairing key
  4303. if (playgroundParticles.playgroundCache.propertyColorId[p] != thisManipulator.manipulatorId) {
  4304. playgroundParticles.playgroundCache.propertyColorId[p] = thisManipulator.manipulatorId;
  4305. }
  4306. break;
  4307. // Size Property
  4308. case MANIPULATORPROPERTYTYPEC.Size:
  4309. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4310. manipulatorDistance = thisManipulator.strengthDistanceEffect>0f?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:0;
  4311. if (manipulatorDistance==0)
  4312. playgroundParticles.playgroundCache.size[p] = thisManipulatorProperty.size;
  4313. else
  4314. playgroundParticles.playgroundCache.size[p] = thisManipulatorProperty.size+(thisManipulatorProperty.size/(1f+(manipulatorDistance/thisManipulator.strengthSmoothing))*(thisManipulator.strength*thisManipulatorProperty.strength));
  4315. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4316. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4317. playgroundParticles.playgroundCache.size[p] = Mathf.Lerp(playgroundParticles.playgroundCache.size[p], thisManipulatorProperty.size, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4318. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4319. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4320. playgroundParticles.playgroundCache.size[p] = Mathf.MoveTowards(playgroundParticles.playgroundCache.size[p], thisManipulatorProperty.size, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4321. }
  4322. playgroundParticles.playgroundCache.changedByPropertySize[p] = true;
  4323. break;
  4324. // Target Property
  4325. case MANIPULATORPROPERTYTYPEC.Target:
  4326. if (thisManipulatorProperty.targets.Count>0 && thisManipulatorProperty.targets[thisManipulatorProperty.targetPointer].available) {
  4327. // Set target pointer
  4328. if (playgroundParticles.playgroundCache.propertyId[p] != thisManipulator.manipulatorId) {
  4329. playgroundParticles.playgroundCache.propertyTarget[p] = thisManipulatorProperty.targetPointer;
  4330. thisManipulatorProperty.targetPointer++; thisManipulatorProperty.targetPointer=thisManipulatorProperty.targetPointer%thisManipulatorProperty.targets.Count;
  4331. playgroundParticles.playgroundCache.propertyId[p] = thisManipulator.manipulatorId;
  4332. }
  4333. // Teleport or lerp to position based on transition type
  4334. if (playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId && thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].available) {
  4335. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None)
  4336. playgroundParticles.playgroundCache.position[p] = localSpace?
  4337. thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].localPosition
  4338. :
  4339. thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].position;
  4340. else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4341. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4342. playgroundParticles.playgroundCache.position[p] = localSpace?
  4343. Vector3.Lerp(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].localPosition, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing)
  4344. :
  4345. Vector3.Lerp(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].position, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4346. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4347. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4348. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4349. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4350. playgroundParticles.playgroundCache.position[p] = localSpace?
  4351. Vector3.MoveTowards(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].localPosition, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing)
  4352. :
  4353. Vector3.MoveTowards(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].position, (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4354. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4355. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4356. }
  4357. // This particle was changed by a target property
  4358. playgroundParticles.playgroundCache.changedByPropertyTarget[p] = true;
  4359. }
  4360. }
  4361. break;
  4362. // Death Property
  4363. case MANIPULATORPROPERTYTYPEC.Death:
  4364. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None)
  4365. playgroundParticles.playgroundCache.life[p] = playgroundParticles.lifetime;
  4366. else {
  4367. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4368. playgroundParticles.playgroundCache.birth[p] -= (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing;
  4369. }
  4370. // This particle was changed by a death property
  4371. playgroundParticles.playgroundCache.changedByPropertyDeath[p] = true;
  4372. break;
  4373. // Attractors
  4374. case MANIPULATORPROPERTYTYPEC.Attractor:
  4375. if (!playgroundParticles.onlySourcePositioning) {
  4376. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4377. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], (manipulatorPosition-particlePosition)*((thisManipulatorProperty.strength*thisManipulator.strength)/manipulatorDistance), t*((thisManipulatorProperty.strength*thisManipulator.strength)/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4378. }
  4379. break;
  4380. // Attractors Gravitational
  4381. case MANIPULATORPROPERTYTYPEC.Gravitational:
  4382. if (!playgroundParticles.onlySourcePositioning) {
  4383. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4384. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], (manipulatorPosition-particlePosition)*(thisManipulatorProperty.strength*thisManipulator.strength)/manipulatorDistance, t/thisManipulator.strengthSmoothing);
  4385. }
  4386. break;
  4387. // Repellents
  4388. case MANIPULATORPROPERTYTYPEC.Repellent:
  4389. if (!playgroundParticles.onlySourcePositioning) {
  4390. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4391. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], (particlePosition-manipulatorPosition)*((thisManipulatorProperty.strength*thisManipulator.strength)/manipulatorDistance), t*((thisManipulatorProperty.strength*thisManipulator.strength)/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4392. }
  4393. break;
  4394. // Vortex
  4395. case MANIPULATORPROPERTYTYPEC.Vortex:
  4396. if (!playgroundParticles.onlySourcePositioning) {
  4397. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, playgroundParticles.playgroundCache.position[p])/thisManipulator.strengthDistanceEffect:10f;
  4398. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], ((manipulatorPosition-particlePosition)*(thisManipulator.strength*thisManipulatorProperty.strength)/manipulatorDistance)-Vector3.Cross(thisManipulator.transform.up, (manipulatorPosition-particlePosition))*(thisManipulator.strength*thisManipulatorProperty.strength)/manipulatorDistance, (t*(thisManipulator.strength*thisManipulatorProperty.strength)/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4399. }
  4400. break;
  4401. // Turbulence
  4402. case MANIPULATORPROPERTYTYPEC.Turbulence:
  4403. if (!playgroundParticles.onlySourcePositioning && thisManipulatorProperty.turbulenceType!=TURBULENCETYPE.None) {
  4404. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.Distance (manipulatorPosition, particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4405. Turbulence (
  4406. playgroundParticles,
  4407. thisManipulatorProperty.turbulenceSimplex,
  4408. p,
  4409. t/thisManipulator.strengthSmoothing,
  4410. thisManipulatorProperty.turbulenceType,
  4411. thisManipulatorProperty.turbulenceTimeScale,
  4412. thisManipulatorProperty.turbulenceScale,
  4413. ((thisManipulatorProperty.strength*thisManipulator.strength)/manipulatorDistance),
  4414. thisManipulatorProperty.turbulenceApplyLifetimeStrength,
  4415. thisManipulatorProperty.turbulenceLifetimeStrength
  4416. );
  4417. }
  4418. break;
  4419. // Mesh Target
  4420. case MANIPULATORPROPERTYTYPEC.MeshTarget:
  4421. if (!playgroundParticles.onlySourcePositioning && thisManipulatorProperty.meshTarget.initialized) {
  4422. // Set target pointer
  4423. if (playgroundParticles.playgroundCache.propertyId[p] != thisManipulator.manipulatorId) {
  4424. playgroundParticles.playgroundCache.propertyTarget[p] = thisManipulatorProperty.targetSortingList[thisManipulatorProperty.targetPointer];
  4425. thisManipulatorProperty.targetPointer++; thisManipulatorProperty.targetPointer=thisManipulatorProperty.targetPointer%thisManipulatorProperty.meshTarget.vertexPositions.Length;
  4426. playgroundParticles.playgroundCache.propertyId[p] = thisManipulator.manipulatorId;
  4427. }
  4428. // Teleport or lerp to position based on transition type
  4429. if (playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4430. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4431. playgroundParticles.playgroundCache.position[p] = thisManipulatorProperty.meshTargetMatrix.MultiplyPoint3x4(thisManipulatorProperty.meshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.meshTarget.vertexPositions.Length]);
  4432. playgroundParticles.playgroundCache.velocity[p] = Vector3.zero;
  4433. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4434. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, thisManipulatorProperty.meshTargetMatrix.MultiplyPoint3x4(thisManipulatorProperty.meshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.meshTarget.vertexPositions.Length]), (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4435. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4436. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4437. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4438. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, thisManipulatorProperty.meshTargetMatrix.MultiplyPoint3x4(thisManipulatorProperty.meshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.meshTarget.vertexPositions.Length]), (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4439. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4440. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4441. }
  4442. // This particle was changed by a target property
  4443. playgroundParticles.playgroundCache.changedByPropertyTarget[p] = true;
  4444. }
  4445. }
  4446. break;
  4447. // Skinned Mesh Target
  4448. case MANIPULATORPROPERTYTYPEC.SkinnedMeshTarget:
  4449. if (!playgroundParticles.onlySourcePositioning && thisManipulatorProperty.skinnedMeshTarget.initialized) {
  4450. // Set target pointer
  4451. if (playgroundParticles.playgroundCache.propertyId[p] != thisManipulator.manipulatorId) {
  4452. playgroundParticles.playgroundCache.propertyTarget[p] = thisManipulatorProperty.targetSortingList[thisManipulatorProperty.targetPointer];
  4453. thisManipulatorProperty.targetPointer++; thisManipulatorProperty.targetPointer=thisManipulatorProperty.targetPointer%thisManipulatorProperty.skinnedMeshTarget.vertexPositions.Length;
  4454. playgroundParticles.playgroundCache.propertyId[p] = thisManipulator.manipulatorId;
  4455. }
  4456. // Teleport or lerp to position based on transition type
  4457. if (playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4458. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4459. playgroundParticles.playgroundCache.position[p] = thisManipulatorProperty.skinnedMeshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.skinnedMeshTarget.vertexPositions.Length];
  4460. playgroundParticles.playgroundCache.velocity[p] = Vector3.zero;
  4461. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4462. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4463. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, thisManipulatorProperty.skinnedMeshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.skinnedMeshTarget.vertexPositions.Length], t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4464. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4465. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4466. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4467. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4468. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, thisManipulatorProperty.skinnedMeshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.skinnedMeshTarget.vertexPositions.Length], t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4469. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4470. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4471. }
  4472. // This particle was changed by a target property
  4473. playgroundParticles.playgroundCache.changedByPropertyTarget[p] = true;
  4474. }
  4475. }
  4476. break;
  4477. // State Target
  4478. case MANIPULATORPROPERTYTYPEC.StateTarget:
  4479. if (!playgroundParticles.onlySourcePositioning && thisManipulatorProperty.stateTarget.initialized && !thisManipulatorProperty.stateTarget.IsInitializing()) {
  4480. // Set target pointer
  4481. if (playgroundParticles.playgroundCache.propertyId[p] != thisManipulator.manipulatorId) {
  4482. playgroundParticles.playgroundCache.propertyTarget[p] = thisManipulatorProperty.targetSortingList[thisManipulatorProperty.targetPointer%thisManipulatorProperty.targetSortingList.Length];
  4483. thisManipulatorProperty.targetPointer++; thisManipulatorProperty.targetPointer=thisManipulatorProperty.targetPointer%thisManipulatorProperty.stateTarget.positionLength;
  4484. playgroundParticles.playgroundCache.propertyId[p] = thisManipulator.manipulatorId;
  4485. }
  4486. // Teleport or lerp to position based on transition type
  4487. if (playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4488. Vector3 newPos = thisManipulatorProperty.stateTarget.stateTransformMx.MultiplyPoint3x4(thisManipulatorProperty.stateTarget.GetPosition(playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.stateTarget.positionLength));
  4489. if (!thisManipulatorProperty.onlyPositionInRange || thisManipulatorProperty.onlyPositionInRange && thisManipulator.Contains (newPos, manipulatorPosition)) {
  4490. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4491. playgroundParticles.playgroundCache.position[p] = newPos;
  4492. playgroundParticles.playgroundCache.velocity[p] = Vector3.zero;
  4493. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4494. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4495. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, newPos, t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4496. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4497. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4498. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4499. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4500. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, newPos, t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4501. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4502. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4503. }
  4504. // This particle was changed by a target property
  4505. playgroundParticles.playgroundCache.changedByPropertyTarget[p] = true;
  4506. }
  4507. }
  4508. }
  4509. break;
  4510. // Spline Target
  4511. case MANIPULATORPROPERTYTYPEC.SplineTarget:
  4512. if (!playgroundParticles.onlySourcePositioning && thisManipulatorProperty.SplineTargetIsReady()) {
  4513. // Set target pointer
  4514. if (playgroundParticles.playgroundCache.propertyId[p] != thisManipulator.manipulatorId) {
  4515. playgroundParticles.playgroundCache.propertyId[p] = thisManipulator.manipulatorId;
  4516. }
  4517. // Teleport or lerp to position based on transition type
  4518. if (playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4519. Vector3 newPos = thisManipulatorProperty.splineTarget.GetPoint(thisManipulatorProperty.splineTargetMethod==SPLINETARGETMETHOD.SplineTime? ((p*1f)/(playgroundParticles.particleCount*1f))+thisManipulatorProperty.splineTimeOffset : ((playgroundParticles.playgroundCache.life[p]+playgroundParticles.playgroundCache.lifetimeSubtraction[p])/playgroundParticles.lifetime)+thisManipulatorProperty.splineTimeOffset);
  4520. if (!thisManipulatorProperty.onlyPositionInRange || thisManipulatorProperty.onlyPositionInRange && thisManipulator.Contains (newPos, manipulatorPosition)) {
  4521. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4522. playgroundParticles.playgroundCache.position[p] = newPos;
  4523. playgroundParticles.playgroundCache.velocity[p] = Vector3.zero;
  4524. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4525. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4526. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, newPos, t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4527. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4528. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4529. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4530. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4531. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, newPos, t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4532. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4533. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4534. }
  4535. // This particle was changed by a target property
  4536. playgroundParticles.playgroundCache.changedByPropertyTarget[p] = true;
  4537. }
  4538. }
  4539. }
  4540. break;
  4541. }
  4542. playgroundParticles.playgroundCache.changedByProperty[p] = true;
  4543. } else {
  4544. // Handle size outside
  4545. if (thisManipulatorProperty.onlySizeInRange)
  4546. playgroundParticles.playgroundCache.changedByPropertySize[p] = false;
  4547. // Handle colors outside of property manipulator range
  4548. if (playgroundParticles.playgroundCache.propertyColorId[p] == thisManipulator.manipulatorId && (thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.Color || thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.LifetimeColor)) {
  4549. // Lerp back color with previous set key
  4550. if (playgroundParticles.playgroundCache.changedByPropertyColorLerp[p] && thisManipulatorProperty.transition != MANIPULATORPROPERTYTRANSITIONC.None && thisManipulatorProperty.onlyColorInRange) {
  4551. playgroundParticles.particleCache[p].color = Color.Lerp(playgroundParticles.particleCache[p].color, playgroundParticles.lifetimeColor.Evaluate(playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime), t*thisManipulatorProperty.strength*thisManipulator.strength);
  4552. }
  4553. if (thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.LifetimeColor && !thisManipulatorProperty.onlyColorInRange) {
  4554. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.None) {
  4555. playgroundParticles.particleCache[p].color = thisManipulatorProperty.lifetimeColor.Evaluate(playgroundParticles.lifetime>0?playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime:0);
  4556. } else {
  4557. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4558. playgroundParticles.particleCache[p].color = Color.Lerp(playgroundParticles.particleCache[p].color, thisManipulatorProperty.lifetimeColor.Evaluate(playgroundParticles.lifetime>0?playgroundParticles.playgroundCache.life[p]/playgroundParticles.lifetime:0), (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4559. }
  4560. }
  4561. if (thisManipulatorProperty.onlyColorInRange)
  4562. playgroundParticles.playgroundCache.changedByPropertyColor[p] = false;
  4563. }
  4564. // Position onto targets when outside of range
  4565. if (!playgroundParticles.onlySourcePositioning && !thisManipulatorProperty.onlyPositionInRange && thisManipulatorProperty.transition != MANIPULATORPROPERTYTRANSITIONC.None) {
  4566. // Target (Transform) positioning outside of range
  4567. if (thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.Target) {
  4568. if (thisManipulatorProperty.targets.Count>0 && thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count]!=null) {
  4569. if (playgroundParticles.playgroundCache.changedByPropertyTarget[p] && thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].available && playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.transform.GetInstanceID()) {
  4570. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp)
  4571. playgroundParticles.playgroundCache.position[p] = localSpace?
  4572. Vector3.Lerp(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].localPosition, t*(thisManipulatorProperty.strength*thisManipulator.strength)/thisManipulator.strengthSmoothing)
  4573. :
  4574. Vector3.Lerp(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].position, t*(thisManipulatorProperty.strength*thisManipulator.strength)/thisManipulator.strengthSmoothing);
  4575. else
  4576. playgroundParticles.playgroundCache.position[p] = localSpace?
  4577. Vector3.MoveTowards(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].localPosition, t*(thisManipulatorProperty.strength*thisManipulator.strength)/thisManipulator.strengthSmoothing)
  4578. :
  4579. Vector3.MoveTowards(particlePosition, thisManipulatorProperty.targets[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.targets.Count].position, t*(thisManipulatorProperty.strength*thisManipulator.strength)/thisManipulator.strengthSmoothing);
  4580. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4581. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4582. }
  4583. }
  4584. }
  4585. // Mesh Target positioning outside of range
  4586. if (thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.MeshTarget && thisManipulatorProperty.meshTarget.initialized && playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4587. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4588. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4589. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, thisManipulatorProperty.meshTargetMatrix.MultiplyPoint3x4(thisManipulatorProperty.meshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.meshTarget.vertexPositions.Length]), (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4590. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4591. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4592. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4593. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4594. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, thisManipulatorProperty.meshTargetMatrix.MultiplyPoint3x4(thisManipulatorProperty.meshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.meshTarget.vertexPositions.Length]), (t*thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4595. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4596. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4597. }
  4598. }
  4599. // Skinned Mesh Target positioning outside of range
  4600. if (thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.SkinnedMeshTarget && thisManipulatorProperty.skinnedMeshTarget.initialized && playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4601. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4602. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4603. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, thisManipulatorProperty.skinnedMeshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.skinnedMeshTarget.vertexPositions.Length], t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4604. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4605. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4606. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4607. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4608. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, thisManipulatorProperty.skinnedMeshTarget.vertexPositions[playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.skinnedMeshTarget.vertexPositions.Length], t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4609. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4610. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4611. }
  4612. }
  4613. // State Target positioning outside of range
  4614. if (thisManipulatorProperty.type == MANIPULATORPROPERTYTYPEC.StateTarget && thisManipulatorProperty.stateTarget.initialized && !thisManipulatorProperty.stateTarget.IsInitializing() && playgroundParticles.playgroundCache.propertyId[p] == thisManipulator.manipulatorId) {
  4615. if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Lerp) {
  4616. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4617. playgroundParticles.playgroundCache.position[p] = Vector3.Lerp(particlePosition, thisManipulatorProperty.stateTarget.stateTransformMx.MultiplyPoint3x4(thisManipulatorProperty.stateTarget.GetPosition(playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.stateTarget.positionLength)), t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4618. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4619. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4620. } else if (thisManipulatorProperty.transition == MANIPULATORPROPERTYTRANSITIONC.Linear) {
  4621. manipulatorDistance = thisManipulator.strengthDistanceEffect>0?Vector3.SqrMagnitude (manipulatorPosition - particlePosition)/thisManipulator.strengthDistanceEffect:10f;
  4622. playgroundParticles.playgroundCache.position[p] = Vector3.MoveTowards(particlePosition, thisManipulatorProperty.stateTarget.stateTransformMx.MultiplyPoint3x4(thisManipulatorProperty.stateTarget.GetPosition(playgroundParticles.playgroundCache.propertyTarget[p]%thisManipulatorProperty.stateTarget.positionLength)), t*(thisManipulatorProperty.strength*thisManipulator.strength/manipulatorDistance)/thisManipulator.strengthSmoothing);
  4623. if (thisManipulatorProperty.zeroVelocityStrength>0)
  4624. playgroundParticles.playgroundCache.velocity[p] = Vector3.Lerp(playgroundParticles.playgroundCache.velocity[p], Vector3.zero, t*thisManipulatorProperty.zeroVelocityStrength);
  4625. }
  4626. }
  4627. }
  4628. }
  4629. }
  4630. // Update the source scatter
  4631. public void RefreshScatter () {
  4632. System.Random random = new System.Random();
  4633. for (int p = 0; p<particleCount; p++) {
  4634. if (applySourceScatter) {
  4635. if (sourceScatterMethod==MINMAXVECTOR3METHOD.Rectangular)
  4636. playgroundCache.scatterPosition[p] = RandomRange(random, sourceScatterMin, sourceScatterMax);
  4637. else if (sourceScatterMethod==MINMAXVECTOR3METHOD.RectangularLinear)
  4638. playgroundCache.scatterPosition[p] = Vector3.Lerp (sourceScatterMin, sourceScatterMax, (p*1f)/(particleCount*1f));
  4639. else if (sourceScatterMethod==MINMAXVECTOR3METHOD.Spherical)
  4640. playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, sourceScatterMin.x, sourceScatterMax.x);
  4641. else if (sourceScatterMethod==MINMAXVECTOR3METHOD.SphericalLinear)
  4642. playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, sourceScatterMin.x, sourceScatterMax.x, (p*1f)/(particleCount*1f));
  4643. //else if (sourceScatterMethod==MINMAXVECTOR3METHOD.SphericalSector)
  4644. // playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, sourceScatterMin.x, sourceScatterMax.x, sourceScatterMin.y, sourceScatterMax.y);
  4645. //else if (sourceScatterMethod==MINMAXVECTOR3METHOD.SphericalSectorLinear)
  4646. // playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, sourceScatterMin.x, sourceScatterMax.x, sourceScatterMin.y, sourceScatterMax.y, (p*1f)/(particleCount*1f));
  4647. } else playgroundCache.scatterPosition[p] = Vector3.zero;
  4648. }
  4649. }
  4650. /// <summary>
  4651. /// Returns a random range float, adapted for threading.
  4652. /// </summary>
  4653. /// <returns>The range.</returns>
  4654. /// <param name="random">Random.</param>
  4655. /// <param name="min">Minimum.</param>
  4656. /// <param name="max">Max.</param>
  4657. public static float RandomRange (System.Random random, float min, float max) {
  4658. return min+((float)random.NextDouble())*(max-min);
  4659. }
  4660. /// <summary>
  4661. /// Random range Vector3 adapted for threading. This produces a rectangular random position inside min to max. Should you want spherical distribution use RandomRangeSpherical instead.
  4662. /// </summary>
  4663. /// <returns>The range.</returns>
  4664. /// <param name="random">Random.</param>
  4665. /// <param name="min">Minimum.</param>
  4666. /// <param name="max">Max.</param>
  4667. public static Vector3 RandomRange (System.Random random, Vector3 min, Vector3 max) {
  4668. return new Vector3 (
  4669. RandomRange(random, min.x, max.x),
  4670. RandomRange(random, min.y, max.y),
  4671. RandomRange(random, min.z, max.z)
  4672. );
  4673. }
  4674. /// <summary>
  4675. /// Spherical random range Vector3, adapted for threading
  4676. /// </summary>
  4677. /// <returns>The spherical random range.</returns>
  4678. /// <param name="random">Random.</param>
  4679. /// <param name="min">Minimum.</param>
  4680. /// <param name="max">Max.</param>
  4681. public static Vector3 RandomRangeSpherical (System.Random random, float min, float max) {
  4682. float a = 2f*Mathf.PI*(RandomRange(random, 0f, 1f));
  4683. float b = Mathf.Asin (2f*(RandomRange(random, 0f, 1f))-1f);
  4684. float r = RandomRange(random, min, max);
  4685. return new Vector3 (
  4686. Mathf.Cos(b)*Mathf.Sin(a)*r,
  4687. Mathf.Cos(b)*Mathf.Cos(a)*r,
  4688. Mathf.Sin(b)*r
  4689. );
  4690. }
  4691. // Spherical random range Vector3 adapted for threading
  4692. public static Vector3 RandomRangeSpherical (System.Random random, float min, float max, float step) {
  4693. float a = 2f*Mathf.PI*(RandomRange(random, 0f, 1f));
  4694. float b = Mathf.Asin (2f*(RandomRange(random, 0f, 1f))-1f);
  4695. float r = Mathf.Lerp (min, max, step);
  4696. return new Vector3 (
  4697. Mathf.Cos(b)*Mathf.Sin(a)*r,
  4698. Mathf.Cos(b)*Mathf.Cos(a)*r,
  4699. Mathf.Sin(b)*r
  4700. );
  4701. }
  4702. // Spherical random range Vector3 with partial sector option adapted for threading
  4703. public static Vector3 RandomRangeSpherical (System.Random random, float min, float max, float sectorA, float sectorB) {
  4704. if (sectorB<0) sectorB = 0;
  4705. float a = (2f*sectorA)*Mathf.PI*(RandomRange(random, 0f, 1f));
  4706. float b = Mathf.Asin ((2f*sectorB)*(RandomRange(random, 0f, 1f))-1f);
  4707. float r = RandomRange(random, min, max);
  4708. return new Vector3 (
  4709. Mathf.Cos(b)*Mathf.Sin(a)*r,
  4710. Mathf.Cos(b)*Mathf.Cos(a)*r,
  4711. Mathf.Sin(b)*r
  4712. );
  4713. }
  4714. // Spherical random range Vector3 with partial sector option adapted for threading
  4715. public static Vector3 RandomRangeSpherical (System.Random random, float min, float max, float sectorA, float sectorB, float step) {
  4716. if (sectorB<0) sectorB = 0;
  4717. float a = (2f*sectorA)*Mathf.PI*(RandomRange(random, 0f, 1f));
  4718. float b = Mathf.Asin ((2f*sectorB)*(RandomRange(random, 0f, 1f))-1f);
  4719. float r = Mathf.Lerp (min, max, step);
  4720. return new Vector3 (
  4721. Mathf.Cos(b)*Mathf.Sin(a)*r,
  4722. Mathf.Cos(b)*Mathf.Cos(a)*r,
  4723. Mathf.Sin(b)*r
  4724. );
  4725. }
  4726. // Return a random float array
  4727. public static float[] RandomFloat (int length, float min, float max, System.Random random) {
  4728. float[] f = new float[length];
  4729. for (int i = 0; i<length; i++) {
  4730. f[i] = RandomRange (random, min, max);
  4731. }
  4732. return f;
  4733. }
  4734. public void InactivateParticle (int particleId) {
  4735. playgroundCache.simulate[particleId] = false;
  4736. playgroundCache.rebirth[particleId] = false;
  4737. playgroundCache.isNonBirthed[particleId] = true;
  4738. playgroundCache.isFirstLoop[particleId] = true;
  4739. playgroundCache.isCalculatedThisFrame[particleId] = false;
  4740. playgroundCache.position[particleId] = PlaygroundC.initialTargetPosition;
  4741. playgroundCache.targetPosition[particleId] = playgroundCache.position[particleId];
  4742. playgroundCache.previousTargetPosition[particleId] = playgroundCache.targetPosition[particleId];
  4743. playgroundCache.velocity[particleId] = Vector3.zero;
  4744. particleCache[particleId].position = playgroundCache.position[particleId];
  4745. particleCache[particleId].size = 0;
  4746. // Reset manipulators influence
  4747. playgroundCache.changedByProperty[particleId] = false;
  4748. playgroundCache.changedByPropertyColor[particleId] = false;
  4749. playgroundCache.changedByPropertyColorLerp[particleId] = false;
  4750. playgroundCache.changedByPropertyColorKeepAlpha[particleId] = false;
  4751. playgroundCache.changedByPropertySize[particleId] = false;
  4752. playgroundCache.changedByPropertyTarget[particleId] = false;
  4753. playgroundCache.changedByPropertyDeath[particleId] = false;
  4754. playgroundCache.propertyTarget[particleId] = 0;
  4755. playgroundCache.propertyId[particleId] = 0;
  4756. playgroundCache.propertyColorId[particleId] = 0;
  4757. playgroundCache.manipulatorId[particleId] = 0;
  4758. }
  4759. public void InactivateParticles () {
  4760. for (int p = 0; p<particleCount; p++)
  4761. InactivateParticle(p);
  4762. hasActiveParticles = false;
  4763. threadHadNoActiveParticles = true;
  4764. isDoneThread = true;
  4765. if (particleCache!=null && particleCache.Length>0)
  4766. shurikenParticleSystem.SetParticles(particleCache, particleCache.Length);
  4767. }
  4768. // Rebirth of a specified particle
  4769. public static void Rebirth (PlaygroundParticlesC playgroundParticles, int p, System.Random random) {
  4770. if (!playgroundParticles.hasActiveParticles) return;
  4771. // Set initial values
  4772. playgroundParticles.playgroundCache.rebirth[p] = playgroundParticles.source==SOURCEC.Script?true:(playgroundParticles.emit && (playgroundParticles.loop || playgroundParticles.playgroundCache.isNonBirthed[p]) && playgroundParticles.playgroundCache.emission[p]);
  4773. playgroundParticles.playgroundCache.isFirstLoop[p] = playgroundParticles.playgroundCache.isNonBirthed[p];
  4774. playgroundParticles.playgroundCache.isNonBirthed[p] = false;
  4775. playgroundParticles.playgroundCache.life[p] = 0f;
  4776. playgroundParticles.playgroundCache.birth[p] = playgroundParticles.playgroundCache.death[p];
  4777. playgroundParticles.playgroundCache.death[p] += playgroundParticles.lifetime;
  4778. playgroundParticles.playgroundCache.velocity[p] = Vector3.zero;
  4779. playgroundParticles.playgroundCache.noForce[p] = false;
  4780. if (playgroundParticles.hasCollisionCache)
  4781. playgroundParticles.collisionCache.Reset(p);
  4782. // Reset manipulators influence
  4783. playgroundParticles.playgroundCache.changedByProperty[p] = false;
  4784. playgroundParticles.playgroundCache.changedByPropertyColor[p] = false;
  4785. playgroundParticles.playgroundCache.changedByPropertyColorLerp[p] = false;
  4786. playgroundParticles.playgroundCache.changedByPropertyColorKeepAlpha[p] = false;
  4787. playgroundParticles.playgroundCache.changedByPropertySize[p] = false;
  4788. playgroundParticles.playgroundCache.changedByPropertyTarget[p] = false;
  4789. playgroundParticles.playgroundCache.changedByPropertyDeath[p] = false;
  4790. playgroundParticles.playgroundCache.propertyTarget[p] = 0;
  4791. playgroundParticles.playgroundCache.propertyId[p] = 0;
  4792. playgroundParticles.playgroundCache.propertyColorId[p] = 0;
  4793. playgroundParticles.playgroundCache.manipulatorId[p] = 0;
  4794. // Set new random size
  4795. if (playgroundParticles.applyRandomSizeOnRebirth)
  4796. playgroundParticles.playgroundCache.initialSize[p] = RandomRange(random, playgroundParticles.sizeMin, playgroundParticles.sizeMax);
  4797. // Initial velocity
  4798. if (!playgroundParticles.onlySourcePositioning && !playgroundParticles.onlyLifetimePositioning) {
  4799. // Initial global velocity
  4800. if (playgroundParticles.applyInitialVelocity) {
  4801. if (playgroundParticles.applyRandomInitialVelocityOnRebirth) {
  4802. if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.Spherical)
  4803. playgroundParticles.playgroundCache.initialVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialVelocityMin.x, playgroundParticles.initialVelocityMax.x);
  4804. else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.SphericalLinear)
  4805. playgroundParticles.playgroundCache.initialVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialVelocityMin.x, playgroundParticles.initialVelocityMax.x, (p*1f)/(playgroundParticles.particleCount*1f));
  4806. else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.RectangularLinear)
  4807. playgroundParticles.playgroundCache.initialVelocity[p] = Vector3.Lerp (playgroundParticles.initialVelocityMin, playgroundParticles.initialVelocityMax, (p*1f)/(playgroundParticles.particleCount*1f));
  4808. //else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.SphericalSector)
  4809. // playgroundParticles.playgroundCache.initialVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialVelocityMin.x, playgroundParticles.initialVelocityMax.x, playgroundParticles.initialVelocityMin.y, playgroundParticles.initialVelocityMax.y);
  4810. //else if (playgroundParticles.initialVelocityMethod==MINMAXVECTOR3METHOD.SphericalSectorLinear)
  4811. // playgroundParticles.playgroundCache.initialVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialVelocityMin.x, playgroundParticles.initialVelocityMax.x, playgroundParticles.initialVelocityMin.y, playgroundParticles.initialVelocityMax.y, (p*1f)/(playgroundParticles.particleCount*1f));
  4812. else playgroundParticles.playgroundCache.initialVelocity[p] = RandomRange(random, playgroundParticles.initialVelocityMin, playgroundParticles.initialVelocityMax);
  4813. }
  4814. playgroundParticles.playgroundCache.velocity[p] = playgroundParticles.playgroundCache.initialVelocity[p];
  4815. // Give this spawning particle its velocity shape
  4816. if (playgroundParticles.applyInitialVelocityShape)
  4817. playgroundParticles.playgroundCache.velocity[p] = Vector3.Scale(playgroundParticles.playgroundCache.velocity[p], playgroundParticles.initialVelocityShape.Evaluate((p*1f)/(playgroundParticles.particleCount*1f), playgroundParticles.initialVelocityShapeScale));
  4818. }
  4819. // Initial local velocity
  4820. if (playgroundParticles.applyInitialLocalVelocity && playgroundParticles.source!=SOURCEC.Script) {
  4821. if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.Spherical)
  4822. playgroundParticles.playgroundCache.initialLocalVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialLocalVelocityMin.x, playgroundParticles.initialLocalVelocityMax.x);
  4823. else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.SphericalLinear)
  4824. playgroundParticles.playgroundCache.initialLocalVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialLocalVelocityMin.x, playgroundParticles.initialLocalVelocityMax.x, (p*1f)/(playgroundParticles.particleCount*1f));
  4825. else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.RectangularLinear)
  4826. playgroundParticles.playgroundCache.initialLocalVelocity[p] = Vector3.Lerp (playgroundParticles.initialLocalVelocityMin, playgroundParticles.initialLocalVelocityMax, (p*1f)/(playgroundParticles.particleCount*1f));
  4827. //else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.SphericalSector)
  4828. // playgroundParticles.playgroundCache.initialLocalVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialLocalVelocityMin.x, playgroundParticles.initialLocalVelocityMax.x, playgroundParticles.initialLocalVelocityMin.y, playgroundParticles.initialLocalVelocityMax.y);
  4829. //else if (playgroundParticles.initialLocalVelocityMethod==MINMAXVECTOR3METHOD.SphericalSectorLinear)
  4830. // playgroundParticles.playgroundCache.initialLocalVelocity[p] = RandomRangeSpherical(random, playgroundParticles.initialLocalVelocityMin.x, playgroundParticles.initialLocalVelocityMax.x, playgroundParticles.initialLocalVelocityMin.y, playgroundParticles.initialLocalVelocityMax.y, (p*1f)/(playgroundParticles.particleCount*1f));
  4831. else playgroundParticles.playgroundCache.initialLocalVelocity[p] = RandomRange(random, playgroundParticles.initialLocalVelocityMin, playgroundParticles.initialLocalVelocityMax);
  4832. playgroundParticles.playgroundCache.velocity[p] += playgroundParticles.playgroundCache.targetDirection[p];
  4833. // Give this spawning particle its local velocity shape
  4834. if (playgroundParticles.applyInitialVelocityShape)
  4835. playgroundParticles.playgroundCache.velocity[p] = Vector3.Scale(playgroundParticles.playgroundCache.velocity[p], playgroundParticles.initialVelocityShape.Evaluate((p*1f)/(playgroundParticles.particleCount*1f), playgroundParticles.initialVelocityShapeScale));
  4836. }
  4837. // Initial stretch
  4838. playgroundParticles.particleCache[p].velocity = playgroundParticles.renderModeStretch&&playgroundParticles.applyStretchStartDirection?playgroundParticles.stretchStartDirection:Vector3.zero;
  4839. }
  4840. if (playgroundParticles.source==SOURCEC.Script) {
  4841. // Velocity for script mode
  4842. if (!playgroundParticles.onlySourcePositioning && !playgroundParticles.onlyLifetimePositioning)
  4843. playgroundParticles.playgroundCache.velocity[p] += playgroundParticles.scriptedEmissionVelocity;
  4844. playgroundParticles.playgroundCache.targetPosition[p] = playgroundParticles.scriptedEmissionPosition;
  4845. playgroundParticles.particleCache[p].velocity = playgroundParticles.renderModeStretch&&playgroundParticles.applyStretchStartDirection?playgroundParticles.stretchStartDirection:Vector3.zero;
  4846. }
  4847. // Set new random lifetime
  4848. if (playgroundParticles.scriptedLifetime==0) {
  4849. if (playgroundParticles.applyRandomLifetimeOnRebirth) {
  4850. if (playgroundParticles.lifetimeValueMethod==VALUEMETHOD.Constant) {
  4851. playgroundParticles.playgroundCache.lifetimeSubtraction[p] = 0;
  4852. } else {
  4853. playgroundParticles.playgroundCache.lifetimeSubtraction[p] = playgroundParticles.lifetime-RandomRange (random, playgroundParticles.lifetimeMin, playgroundParticles.lifetime);
  4854. }
  4855. }
  4856. // Set shuriken particles lifetime
  4857. if (!playgroundParticles.syncPositionsOnMainThread)
  4858. playgroundParticles.particleCache[p].remainingLifetime = playgroundParticles.lifetime;
  4859. playgroundParticles.particleCache[p].startLifetime = playgroundParticles.lifetime-playgroundParticles.playgroundCache.lifetimeSubtraction[p];
  4860. } else {
  4861. playgroundParticles.playgroundCache.lifetimeSubtraction[p] = 0;
  4862. if (!playgroundParticles.syncPositionsOnMainThread)
  4863. playgroundParticles.particleCache[p].remainingLifetime = playgroundParticles.scriptedLifetime;
  4864. playgroundParticles.particleCache[p].startLifetime = playgroundParticles.scriptedLifetime;
  4865. }
  4866. if (playgroundParticles.playgroundCache.rebirth[p]) {
  4867. playgroundParticles.particleCache[p].color = playgroundParticles.GetParticleColor(p, 0f, (p*1f)/(playgroundParticles.particleCount*1f));
  4868. // Source Scattering
  4869. if (playgroundParticles.applySourceScatter && playgroundParticles.source!=SOURCEC.Script) {
  4870. if (playgroundParticles.playgroundCache.scatterPosition[p]==Vector3.zero || playgroundParticles.applyRandomScatterOnRebirth) {
  4871. if (playgroundParticles.sourceScatterMethod==MINMAXVECTOR3METHOD.Rectangular)
  4872. playgroundParticles.playgroundCache.scatterPosition[p] = RandomRange(random, playgroundParticles.sourceScatterMin, playgroundParticles.sourceScatterMax);
  4873. else if (playgroundParticles.sourceScatterMethod==MINMAXVECTOR3METHOD.RectangularLinear)
  4874. playgroundParticles.playgroundCache.scatterPosition[p] = Vector3.Lerp (playgroundParticles.sourceScatterMin, playgroundParticles.sourceScatterMax, (p*1f)/(playgroundParticles.particleCount*1f));
  4875. else if (playgroundParticles.sourceScatterMethod==MINMAXVECTOR3METHOD.SphericalLinear)
  4876. playgroundParticles.playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, playgroundParticles.sourceScatterMin.x, playgroundParticles.sourceScatterMax.x, (p*1f)/(playgroundParticles.particleCount*1f));
  4877. //else if (playgroundParticles.sourceScatterMethod==MINMAXVECTOR3METHOD.SphericalSector)
  4878. // playgroundParticles.playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, playgroundParticles.sourceScatterMin.x, playgroundParticles.sourceScatterMax.x, playgroundParticles.sourceScatterMin.y, playgroundParticles.sourceScatterMax.y);
  4879. //else if (playgroundParticles.sourceScatterMethod==MINMAXVECTOR3METHOD.SphericalSectorLinear)
  4880. // playgroundParticles.playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, playgroundParticles.sourceScatterMin.x, playgroundParticles.sourceScatterMax.x, playgroundParticles.sourceScatterMin.y, playgroundParticles.sourceScatterMax.y, (p*1f)/(playgroundParticles.particleCount*1f));
  4881. else playgroundParticles.playgroundCache.scatterPosition[p] = RandomRangeSpherical(random, playgroundParticles.sourceScatterMin.x, playgroundParticles.sourceScatterMax.x);
  4882. }
  4883. } else playgroundParticles.playgroundCache.scatterPosition[p] = Vector3.zero;
  4884. if (!playgroundParticles.onlyLifetimePositioning) {
  4885. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.targetPosition[p];
  4886. if (!playgroundParticles.syncPositionsOnMainThread)
  4887. playgroundParticles.particleCache[p].position = playgroundParticles.playgroundCache.targetPosition[p];
  4888. playgroundParticles.playgroundCache.previousParticlePosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  4889. playgroundParticles.playgroundCache.collisionParticlePosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  4890. } else if (!playgroundParticles.onlySourcePositioning) {
  4891. // Lifetime Positioning by Vector3 Animation Curve
  4892. if (playgroundParticles.lifetimePositioningUsesSourceDirection && playgroundParticles.source!=SOURCEC.Script) {
  4893. playgroundParticles.playgroundCache.position[p] = playgroundParticles.playgroundCache.targetPosition[p]+(playgroundParticles.playgroundCache.targetDirection[p]);
  4894. } else {
  4895. if (!playgroundParticles.applyLifetimePositioningPositionScale) {
  4896. playgroundParticles.playgroundCache.position[p] =
  4897. playgroundParticles.playgroundCache.targetPosition[p]+
  4898. playgroundParticles.lifetimePositioning.Evaluate(0, playgroundParticles.lifetimePositioningScale);
  4899. } else {
  4900. playgroundParticles.playgroundCache.position[p] =
  4901. playgroundParticles.playgroundCache.targetPosition[p]+
  4902. playgroundParticles.lifetimePositioning.Evaluate(0, playgroundParticles.lifetimePositioningScale)*
  4903. playgroundParticles.lifetimePositioningPositionScale.Evaluate(0);
  4904. }
  4905. }
  4906. if (!playgroundParticles.syncPositionsOnMainThread)
  4907. playgroundParticles.particleCache[p].position = playgroundParticles.playgroundCache.targetPosition[p];
  4908. playgroundParticles.playgroundCache.previousParticlePosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  4909. playgroundParticles.playgroundCache.collisionParticlePosition[p] = playgroundParticles.playgroundCache.targetPosition[p];
  4910. }
  4911. if (playgroundParticles.applyInitialColorOnRebirth) {
  4912. playgroundParticles.particleCache[p].color = playgroundParticles.playgroundCache.initialColor[p];
  4913. playgroundParticles.playgroundCache.color[p] = playgroundParticles.playgroundCache.initialColor[p];
  4914. }
  4915. } else {
  4916. playgroundParticles.particleCache[p].position = PlaygroundC.initialTargetPosition;
  4917. }
  4918. // Set new random rotation
  4919. if (playgroundParticles.applyRandomRotationOnRebirth && !playgroundParticles.rotateTowardsDirection)
  4920. playgroundParticles.playgroundCache.initialRotation[p] = RandomRange(random, playgroundParticles.initialRotationMin, playgroundParticles.initialRotationMax);
  4921. if (!playgroundParticles.rotateTowardsDirection)
  4922. playgroundParticles.playgroundCache.rotation[p] = playgroundParticles.playgroundCache.initialRotation[p];
  4923. else {
  4924. Vector3 particleDir;
  4925. if (!playgroundParticles.onlySourcePositioning&&playgroundParticles.onlyLifetimePositioning)
  4926. particleDir = (playgroundParticles.playgroundCache.position[p]+playgroundParticles.lifetimePositioning.Evaluate(.01f, playgroundParticles.lifetimePositioningScale))-playgroundParticles.playgroundCache.position[p];
  4927. else
  4928. particleDir = playgroundParticles.playgroundCache.velocity[p];
  4929. playgroundParticles.playgroundCache.rotation[p] = playgroundParticles.playgroundCache.initialRotation[p]+SignedAngle(
  4930. Vector3.up,
  4931. particleDir,
  4932. playgroundParticles.rotationNormal
  4933. );
  4934. }
  4935. if (!playgroundParticles.syncPositionsOnMainThread)
  4936. playgroundParticles.particleCache[p].rotation = playgroundParticles.playgroundCache.rotation[p];
  4937. // Set size
  4938. if (playgroundParticles.applyLifetimeSize && !playgroundParticles.applyParticleArraySize)
  4939. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.lifetimeSize.Evaluate(0)*playgroundParticles.scale;
  4940. else if (playgroundParticles.applyLifetimeSize && playgroundParticles.applyParticleArraySize)
  4941. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.lifetimeSize.Evaluate(0)*playgroundParticles.particleArraySize.Evaluate((p*1f)/(playgroundParticles.particleCount*1f))*playgroundParticles.scale;
  4942. else if (playgroundParticles.applyParticleArraySize)
  4943. playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.particleArraySize.Evaluate((p*1f)/(playgroundParticles.particleCount*1f))*playgroundParticles.scale;
  4944. else playgroundParticles.playgroundCache.size[p] = playgroundParticles.playgroundCache.initialSize[p]*playgroundParticles.scale;
  4945. if (!playgroundParticles.syncPositionsOnMainThread)
  4946. playgroundParticles.particleCache[p].size = playgroundParticles.playgroundCache.maskAlpha[p]>0?playgroundParticles.playgroundCache.size[p]:0;
  4947. // Set color gradient id
  4948. if (playgroundParticles.colorSource==COLORSOURCEC.LifetimeColors && playgroundParticles.lifetimeColors.Count>0) {
  4949. playgroundParticles.lifetimeColorId++;playgroundParticles.lifetimeColorId=playgroundParticles.lifetimeColorId%playgroundParticles.lifetimeColors.Count;
  4950. playgroundParticles.playgroundCache.lifetimeColorId[p] = playgroundParticles.lifetimeColorId;
  4951. }
  4952. // Local Manipulators
  4953. if (playgroundParticles.calculateManipulatorOnRebirth) {
  4954. for (int m = 0; m<playgroundParticles.manipulators.Count; m++) {
  4955. if (playgroundParticles.manipulators[m].transform!=null) {
  4956. CalculateManipulator(playgroundParticles, playgroundParticles.manipulators[m], p, playgroundParticles.t, playgroundParticles.playgroundCache.life[p], playgroundParticles.playgroundCache.position[p], (playgroundParticles.localSpace?playgroundParticles.manipulators[m].transform.localPosition:playgroundParticles.manipulators[m].transform.position)+playgroundParticles.manipulatorFix, playgroundParticles.localSpace);
  4957. }
  4958. }
  4959. }
  4960. // Send birth event
  4961. if (playgroundParticles.events.Count>0 && playgroundParticles.playgroundCache.rebirth[p])
  4962. playgroundParticles.SendEvent(EVENTTYPEC.Birth, p);
  4963. if (playgroundParticles.hasEventManipulatorLocal) {
  4964. for (int i = 0; i<playgroundParticles.manipulators.Count; i++) {
  4965. if (playgroundParticles.manipulators[i].trackParticles &&
  4966. playgroundParticles.manipulators[i].sendEventBirth &&
  4967. playgroundParticles.manipulators[i].Contains (playgroundParticles.playgroundCache.targetPosition[p], playgroundParticles.manipulators[i].transform.position)) {
  4968. playgroundParticles.UpdateEventParticle(playgroundParticles.manipulators[i].manipulatorEventParticle, p);
  4969. playgroundParticles.manipulators[i].SendParticleEventBirth();
  4970. }
  4971. }
  4972. }
  4973. if (playgroundParticles.hasEventManipulatorGlobal) {
  4974. for (int i = 0; i<PlaygroundC.reference.manipulators.Count; i++) {
  4975. if (PlaygroundC.reference.manipulators[i].trackParticles &&
  4976. PlaygroundC.reference.manipulators[i].sendEventBirth &&
  4977. PlaygroundC.reference.manipulators[i].Contains (playgroundParticles.playgroundCache.targetPosition[p], PlaygroundC.reference.manipulators[i].transform.position)) {
  4978. playgroundParticles.UpdateEventParticle(PlaygroundC.reference.manipulators[i].manipulatorEventParticle, p);
  4979. PlaygroundC.reference.manipulators[i].SendParticleEventBirth();
  4980. }
  4981. }
  4982. }
  4983. }
  4984. // Sends an event to Event- Targets and Listeners
  4985. void SendEvent (EVENTTYPEC eventType, int p) {
  4986. SendEvent(eventType, p, playgroundCache.velocity[p], null, null, null);
  4987. }
  4988. void SendEvent (EVENTTYPEC eventType, int p, Vector3 preEventVelocity) {
  4989. SendEvent(eventType, p, preEventVelocity, null, null, null);
  4990. }
  4991. void SendEvent (EVENTTYPEC eventType, int p, Vector3 preEventVelocity, Transform collisionTransform) {
  4992. SendEvent(eventType, p, preEventVelocity, collisionTransform, null, null);
  4993. }
  4994. void SendEvent (EVENTTYPEC eventType, int p, Vector3 preEventVelocity, Transform collisionTransform, Collider collisionCollider) {
  4995. SendEvent(eventType, p, preEventVelocity, collisionTransform, collisionCollider, null);
  4996. }
  4997. void SendEvent (EVENTTYPEC eventType, int p, Vector3 preEventVelocity, Transform collisionTransform, Collider2D collisionCollider2D) {
  4998. SendEvent(eventType, p, preEventVelocity, collisionTransform, null, collisionCollider2D);
  4999. }
  5000. void SendEvent (EVENTTYPEC eventType, int p, Vector3 preEventVelocity, Transform collisionTransform, Collider collisionCollider, Collider2D collisionCollider2D) {
  5001. Vector3 eventPosition = Vector3.zero;
  5002. Vector3 eventVelocity = Vector3.zero;
  5003. Color32 eventColor = Color.white;
  5004. // Loop through available events
  5005. for (int i = 0; i<events.Count; i++) {
  5006. if (events[i].enabled && events[i].eventType==eventType) {
  5007. // Check thresholds
  5008. if (events[i].eventType==EVENTTYPEC.Collision)
  5009. if (events[i].collisionThreshold>preEventVelocity.sqrMagnitude)
  5010. return;
  5011. // Set event position
  5012. switch (events[i].eventInheritancePosition) {
  5013. case EVENTINHERITANCEC.User:
  5014. eventPosition = events[i].eventPosition;
  5015. break;
  5016. case EVENTINHERITANCEC.Particle:
  5017. eventPosition = eventType==EVENTTYPEC.Death?playgroundCache.previousParticlePosition[p]:playgroundCache.position[p];
  5018. break;
  5019. case EVENTINHERITANCEC.Source:
  5020. eventPosition = playgroundCache.targetPosition[p];
  5021. break;
  5022. }
  5023. // Set event velocity
  5024. switch (events[i].eventInheritanceVelocity) {
  5025. case EVENTINHERITANCEC.User:
  5026. eventVelocity = events[i].eventVelocity;
  5027. break;
  5028. case EVENTINHERITANCEC.Particle:
  5029. eventVelocity = playgroundCache.velocity[p];
  5030. break;
  5031. case EVENTINHERITANCEC.Source:
  5032. if (applyInitialLocalVelocity)
  5033. eventVelocity = playgroundCache.initialLocalVelocity[p];
  5034. if (applyInitialVelocity)
  5035. eventVelocity += playgroundCache.initialVelocity[p];
  5036. if (applyLifetimeVelocity)
  5037. eventVelocity += lifetimeVelocity.Evaluate(Mathf.Clamp01(playgroundCache.life[p]/lifetime), lifetimeVelocityScale);
  5038. if (applyInitialVelocityShape)
  5039. eventVelocity = Vector3.Scale(eventVelocity, initialVelocityShape.Evaluate((p*1f)/(particleCount*1f), initialVelocityShapeScale));
  5040. break;
  5041. }
  5042. // Apply multiplier
  5043. eventVelocity *= events[i].velocityMultiplier;
  5044. // Set event color
  5045. switch (events[i].eventInheritanceColor) {
  5046. case EVENTINHERITANCEC.User:
  5047. eventColor = events[i].eventColor;
  5048. break;
  5049. case EVENTINHERITANCEC.Particle:
  5050. eventColor = particleCache[p].color;
  5051. break;
  5052. case EVENTINHERITANCEC.Source:
  5053. eventColor = playgroundCache.initialColor[p];
  5054. break;
  5055. }
  5056. // Send the event to any Event Listeners
  5057. if (events[i].broadcastType==EVENTBROADCASTC.EventListeners || events[i].broadcastType==EVENTBROADCASTC.Both) {
  5058. UpdateEventParticle(eventParticle, p);
  5059. eventParticle.collisionCollider = collisionCollider;
  5060. eventParticle.collisionCollider2D = collisionCollider2D;
  5061. eventParticle.collisionTransform = collisionTransform;
  5062. events[i].SendParticleEvent(eventParticle);
  5063. // Send the event to the Playground Manager
  5064. if (events[i].sendToManager) {
  5065. switch (events[i].eventType) {
  5066. case EVENTTYPEC.Birth:
  5067. PlaygroundC.SendParticleEventBirth(eventParticle);
  5068. break;
  5069. case EVENTTYPEC.Death:
  5070. PlaygroundC.SendParticleEventDeath(eventParticle);
  5071. break;
  5072. case EVENTTYPEC.Collision:
  5073. PlaygroundC.SendParticleEventCollision(eventParticle);
  5074. break;
  5075. case EVENTTYPEC.Time:
  5076. PlaygroundC.SendParticleEventTime(eventParticle);
  5077. break;
  5078. }
  5079. }
  5080. }
  5081. // Send the event to target
  5082. if (events[i].initializedTarget && (events[i].broadcastType==EVENTBROADCASTC.Target || events[i].broadcastType==EVENTBROADCASTC.Both)) {
  5083. events[i].target.ThreadSafeEmit(eventPosition, eventVelocity, eventColor);
  5084. }
  5085. }
  5086. }
  5087. }
  5088. public bool UpdateEventParticle (PlaygroundEventParticle eParticle, int p) {
  5089. if (p>=particleCount || p<0) return false;
  5090. eParticle.particleSystemId = particleSystemId;
  5091. eParticle.particleId = p;
  5092. eParticle.birth = playgroundCache.birth[p];
  5093. eParticle.birthDelay = playgroundCache.birthDelay[p];
  5094. eParticle.changedByProperty = playgroundCache.changedByProperty[p];
  5095. eParticle.changedByPropertyColor = playgroundCache.changedByPropertyColor[p];
  5096. eParticle.changedByPropertyColorKeepAlpha = playgroundCache.changedByPropertyColorKeepAlpha[p];
  5097. eParticle.changedByPropertyColorLerp = playgroundCache.changedByPropertyColorLerp[p];
  5098. eParticle.changedByPropertyDeath = playgroundCache.changedByPropertyDeath[p];
  5099. eParticle.changedByPropertySize = playgroundCache.changedByPropertySize[p];
  5100. eParticle.changedByPropertyTarget = playgroundCache.changedByPropertyTarget[p];
  5101. eParticle.collisionParticlePosition = playgroundCache.collisionParticlePosition[p];
  5102. eParticle.color = particleCache[p].color;
  5103. eParticle.scriptedColor = playgroundCache.scriptedColor[p];
  5104. eParticle.death = playgroundCache.death[p];
  5105. eParticle.emission = playgroundCache.emission[p];
  5106. eParticle.initialColor = playgroundCache.initialColor[p];
  5107. eParticle.initialLocalVelocity = playgroundCache.initialLocalVelocity[p];
  5108. eParticle.initialRotation = playgroundCache.initialRotation[p];
  5109. eParticle.initialSize = playgroundCache.initialSize[p];
  5110. eParticle.initialVelocity = playgroundCache.initialVelocity[p];
  5111. eParticle.initialLocalVelocity = playgroundCache.initialLocalVelocity[p];
  5112. eParticle.life = playgroundCache.life[p];
  5113. eParticle.lifetimeColorId = playgroundCache.lifetimeColorId[p];
  5114. eParticle.noForce = playgroundCache.noForce[p];
  5115. eParticle.totalLifetime = (playgroundCache.death[p]-playgroundCache.birth[p])-playgroundCache.lifetimeSubtraction[p];
  5116. eParticle.lifetimeOffset = playgroundCache.lifetimeOffset[p];
  5117. eParticle.localSpaceMovementCompensation = playgroundCache.localSpaceMovementCompensation[p];
  5118. eParticle.position = playgroundCache.position[p];
  5119. eParticle.previousParticlePosition = playgroundCache.previousParticlePosition[p];
  5120. eParticle.previousTargetPosition = playgroundCache.previousTargetPosition[p];
  5121. eParticle.propertyColorId = playgroundCache.propertyColorId[p];
  5122. eParticle.propertyId = playgroundCache.propertyId[p];
  5123. eParticle.excludeFromManipulatorId = playgroundCache.excludeFromManipulatorId[p];
  5124. eParticle.propertyTarget = playgroundCache.propertyTarget[p];
  5125. eParticle.rebirth = playgroundCache.rebirth[p];
  5126. eParticle.rotation = playgroundCache.rotation[p];
  5127. eParticle.rotationSpeed = playgroundCache.rotationSpeed[p];
  5128. eParticle.scatterPosition = playgroundCache.scatterPosition[p];
  5129. eParticle.size = playgroundCache.size[p];
  5130. eParticle.targetDirection = playgroundCache.targetDirection[p];
  5131. eParticle.targetPosition = playgroundCache.targetPosition[p];
  5132. eParticle.velocity = playgroundCache.velocity[p];
  5133. eParticle.isMasked = playgroundCache.isMasked[p];
  5134. eParticle.maskAlpha = playgroundCache.maskAlpha[p];
  5135. eParticle.isFirstLoop = playgroundCache.isFirstLoop[p];
  5136. eParticle.isNonBirthed = playgroundCache.isNonBirthed[p];
  5137. return true;
  5138. }
  5139. // Delete a state from states list
  5140. public void RemoveState (int i) {
  5141. int newState = activeState;
  5142. newState = (newState%states.Count)-1;
  5143. if (newState<0) newState = 0;
  5144. states[newState].Initialize();
  5145. activeState = newState;
  5146. states.RemoveAt(i);
  5147. }
  5148. // Wipe out particles in current PlaygroundParticlesC object
  5149. public static void Clear (PlaygroundParticlesC playgroundParticles) {
  5150. playgroundParticles.inTransition = false;
  5151. playgroundParticles.particleCache = new ParticleSystem.Particle[0];
  5152. playgroundParticles.playgroundCache = null;
  5153. playgroundParticles.shurikenParticleSystem.SetParticles(playgroundParticles.particleCache,0);
  5154. playgroundParticles.shurikenParticleSystem.Clear();
  5155. }
  5156. /// <summary>
  5157. /// Store the current state of a particle system as a Snapshot.
  5158. /// </summary>
  5159. public void Save () {
  5160. if (isSnapshot) {
  5161. Debug.Log("A snapshot can't store snapshot data within itself.", gameObject);
  5162. return;
  5163. }
  5164. StartCoroutine (SaveRoutine ("New Snapshot "+(snapshots.Count+1).ToString()));
  5165. }
  5166. /// <summary>
  5167. /// Store the current state of a particle system as a Snapshot and name it.
  5168. /// </summary>
  5169. /// <param name="saveName">Save name.</param>
  5170. public void Save (string saveName) {
  5171. if (isSnapshot) {
  5172. Debug.Log("A snapshot can't store snapshot data within itself.", gameObject);
  5173. return;
  5174. }
  5175. StartCoroutine (SaveRoutine (saveName));
  5176. }
  5177. bool isSaving = false;
  5178. IEnumerator SaveRoutine (string saveName) {
  5179. isSaving = true;
  5180. PlaygroundSave data = new PlaygroundSave();
  5181. data.settings = PlaygroundC.Particle();
  5182. data.settings.isSnapshot = true;
  5183. yield return null;
  5184. data.Save(this);
  5185. while (data.IsSaving())
  5186. yield return null;
  5187. data.settings.transform.parent = particleSystemTransform;
  5188. data.settings.transform.name = saveName;
  5189. data.settings.timeOfSnapshot = localTime;
  5190. data.name = saveName;
  5191. data.time = localTime;
  5192. data.particleCount = particleCount;
  5193. data.lifetime = lifetime;
  5194. data.version = PlaygroundC.version;
  5195. snapshots.Add (data);
  5196. PlaygroundC.reference.particleSystems.Remove (data.settings);
  5197. #if UNITY_EDITOR
  5198. if (!PlaygroundC.reference.showSnapshotsInHierarchy)
  5199. data.settings.gameObject.hideFlags = HideFlags.HideInHierarchy;
  5200. #endif
  5201. isSaving = false;
  5202. }
  5203. /// <summary>
  5204. /// Load from a saved data state (Snapshot) using an int.
  5205. /// </summary>
  5206. /// <param name="loadPointer">Load pointer.</param>
  5207. public void Load (int loadPointer) {
  5208. if (snapshots.Count>0) {
  5209. loadPointer = loadPointer%snapshots.Count;
  5210. StartCoroutine(LoadRoutine(loadPointer, 0));
  5211. } else {
  5212. Debug.Log ("No data to load from. Please use PlaygroundParticlesC.Save() to store a particle system's current state.", particleSystemGameObject);
  5213. }
  5214. }
  5215. /// <summary>
  5216. /// Load from a saved data state (Snapshot) using a string.
  5217. /// </summary>
  5218. /// <param name="loadName">Load name.</param>
  5219. public void Load (string loadName) {
  5220. if (snapshots.Count>0) {
  5221. for (int i = 0; i<snapshots.Count; i++) {
  5222. if (snapshots[i].name == loadName.Trim()) {
  5223. StartCoroutine(LoadRoutine(i, 0));
  5224. return;
  5225. }
  5226. }
  5227. } else {
  5228. Debug.Log ("No data found with the name "+loadName+".", particleSystemGameObject);
  5229. }
  5230. }
  5231. /// <summary>
  5232. /// Load from a saved data state (Snapshot) and apply a mask to hide specified particles.
  5233. /// The mask sorting will be determined by the loading snapshot's mask settings.
  5234. /// </summary>
  5235. /// <param name="loadPointer">The Snapshot you wish to load.</param>
  5236. /// <param name="loadMask">The amount of masked particles in the particle array.</param>
  5237. public void LoadAndApplyMask (int loadPointer, int loadMask) {
  5238. if (snapshots.Count>0) {
  5239. loadPointer = loadPointer%snapshots.Count;
  5240. StartCoroutine(LoadRoutine(loadPointer, loadMask));
  5241. } else {
  5242. Debug.Log ("No data to load from. Please use PlaygroundParticlesC.Save() to store a particle system's current state.", particleSystemGameObject);
  5243. }
  5244. }
  5245. // Snapshot loading routine
  5246. bool isLoading = false;
  5247. bool transitionAvailable = false;
  5248. IEnumerator LoadRoutine (int loadPointer, int mask) {
  5249. if (loadTransition && loadTransitionTime>0 && snapshots[loadPointer].transitionMultiplier>0 && transitionAvailable && !isYieldRefreshing) {
  5250. if (snapshots[loadPointer].loadMaterial && !snapshots[loadPointer].setMaterialAfterTransition && snapshots[loadPointer].settings.particleSystemRenderer.sharedMaterial!=null)
  5251. particleSystemRenderer.sharedMaterial = snapshots[loadPointer].settings.particleSystemRenderer.sharedMaterial;
  5252. if (mask>0) {
  5253. mask = Mathf.Clamp (mask, 0, snapshots[loadPointer].settings.particleCount);
  5254. if (snapshots[loadPointer].settings.snapshotData.isMasked==null || snapshots[loadPointer].settings.snapshotData.isMasked.Length!=snapshots[loadPointer].settings.particleCount) {
  5255. snapshots[loadPointer].settings.snapshotData.isMasked = new bool[snapshots[loadPointer].settings.particleCount];
  5256. snapshots[loadPointer].settings.snapshotData.maskAlpha = new float[snapshots[loadPointer].settings.particleCount];
  5257. }
  5258. snapshots[loadPointer].settings.applyParticleMask = true;
  5259. snapshots[loadPointer].settings.particleMask = mask;
  5260. snapshots[loadPointer].settings.RefreshMaskSorting();
  5261. snapshots[loadPointer].settings.snapshotData.maskSorting = (int[])snapshots[loadPointer].settings.playgroundCache.maskSorting.Clone();
  5262. for (int i = 0; i<snapshots[loadPointer].settings.particleCount; i++) {
  5263. int maskedP = snapshots[loadPointer].settings.snapshotData.maskSorting[i];
  5264. snapshots[loadPointer].settings.snapshotData.isMasked[maskedP] = i<mask;
  5265. snapshots[loadPointer].settings.snapshotData.maskAlpha[maskedP] = i<mask?0:1f;
  5266. }
  5267. }
  5268. StartCoroutine(LoadTransition(loadPointer));
  5269. while (inTransition) yield return null;
  5270. }
  5271. if (isLoading || inTransition || abortTransition)
  5272. yield break;
  5273. isLoading = true;
  5274. int prevParticleCount = particleCount;
  5275. if (prevParticleCount!=snapshots[loadPointer].settings.particleCount) {
  5276. SetParticleCount(thisInstance, snapshots[loadPointer].settings.particleCount);
  5277. while (isSettingParticleCount)
  5278. yield return null;
  5279. }
  5280. snapshots[loadPointer].Load(thisInstance);
  5281. while (snapshots[loadPointer].IsLoading())
  5282. yield return null;
  5283. lastTimeUpdated = PlaygroundC.globalTime;
  5284. localTime = PlaygroundC.globalTime;
  5285. cameFromNonCalculatedFrame = false;
  5286. cameFromNonEmissionFrame = false;
  5287. if (snapshots[loadPointer].loadMode!=1) {
  5288. if (loadFromStart && isYieldRefreshing)
  5289. yield return null;
  5290. float tos = snapshots[loadPointer].settings.timeOfSnapshot<=0?snapshots[loadPointer].time:snapshots[loadPointer].settings.timeOfSnapshot;
  5291. localTime = tos;
  5292. for (int p = 0; p<particleCount; p++) {
  5293. playgroundCache.birth[p] = tos+(playgroundCache.birth[p]-tos);
  5294. playgroundCache.death[p] = tos+(playgroundCache.death[p]-tos);
  5295. playgroundCache.size[p] = snapshots[loadPointer].settings.snapshotData.size[p];
  5296. particleCache[p].size = playgroundCache.size[p];
  5297. }
  5298. lifetime = snapshots[loadPointer].settings.lifetime;
  5299. previousLifetimeValueMethod = lifetimeValueMethod;
  5300. previousLifetime = lifetime;
  5301. previousLifetimeMin = lifetimeMin;
  5302. previousLifetimeEmission = lifetimeEmission;
  5303. previousNearestNeighborOrigin = nearestNeighborOrigin;
  5304. previousNearestNeighborOriginMethod = nearestNeighborOriginMethod;
  5305. previousNearestNeighborOriginVector3 = nearestNeighborOriginVector3;
  5306. previousNearestNeighborOriginTransform = nearestNeighborOriginTransform;
  5307. previousSorting = sorting;
  5308. localDeltaTime = .001f;
  5309. t = localDeltaTime;
  5310. } else if (prevParticleCount!=particleCount) {
  5311. SetLifetime(thisInstance, sorting, lifetime);
  5312. yield return null;
  5313. while (isSettingLifetime) {
  5314. yield return null;
  5315. }
  5316. }
  5317. hasActiveParticles = true;
  5318. threadHadNoActiveParticles = false;
  5319. lastTimeUpdated = PlaygroundC.globalTime;
  5320. cameFromNonCalculatedFrame = false;
  5321. cameFromNonEmissionFrame = false;
  5322. loopExceeded = false;
  5323. loopExceededOnParticle = -1;
  5324. isLoading = false;
  5325. isDoneThread = true;
  5326. transitionAvailable = true;
  5327. }
  5328. // Apply transition between snapshot load
  5329. bool abortTransition = false;
  5330. IEnumerator LoadTransition (int loadPointer) {
  5331. if (inTransition) {
  5332. abortTransition = true;
  5333. yield return null;
  5334. }
  5335. abortTransition = false;
  5336. inTransition = true;
  5337. float transitionStartTime = PlaygroundC.globalTime;
  5338. float loadParticleLifetime = snapshots[loadPointer].settings.lifetime;
  5339. int loadParticleCount = snapshots[loadPointer].settings.particleCount;
  5340. bool liveParticles = snapshots[loadPointer].loadMode!=1;
  5341. bool firstFrameDone = false;
  5342. int currentParticleCount = particleCount;
  5343. int transitionParticleCount = currentParticleCount;
  5344. float initPosY = PlaygroundC.initialTargetPosition.y;
  5345. PlaygroundCache loadSnapshotData = snapshots[loadPointer].settings.snapshotData;
  5346. Color transparent = new Color(1f,1f,1f,0);
  5347. INDIVIDUALTRANSITIONTYPEC thisSnapshotTransition = snapshots[loadPointer].transitionType;
  5348. if (thisSnapshotTransition==INDIVIDUALTRANSITIONTYPEC.Inherit) {
  5349. thisSnapshotTransition = (INDIVIDUALTRANSITIONTYPEC)((int)loadTransitionType+1);
  5350. }
  5351. // Prepare arrays
  5352. Vector3[] transitionPosition = (Vector3[])playgroundCache.position.Clone();
  5353. Color32[] transitionColor = (Color32[])playgroundCache.color.Clone();
  5354. float[] transitionSize = (float[])playgroundCache.size.Clone();
  5355. float[] transitionRotation = (float[])playgroundCache.rotation.Clone();
  5356. // Resize if more particles are needed
  5357. if (loadParticleCount>particleCount) {
  5358. bool isResizing = true;
  5359. PlaygroundC.RunAsync (()=>{
  5360. System.Array.Resize(ref transitionPosition, loadParticleCount);
  5361. System.Array.Resize(ref transitionColor, loadParticleCount);
  5362. System.Array.Resize(ref transitionSize, loadParticleCount);
  5363. System.Array.Resize(ref transitionRotation, loadParticleCount);
  5364. System.Array.Resize(ref transitionRotation, loadParticleCount);
  5365. System.Array.Resize(ref playgroundCache.position, loadParticleCount);
  5366. System.Array.Resize(ref playgroundCache.color, loadParticleCount);
  5367. System.Array.Resize(ref playgroundCache.size, loadParticleCount);
  5368. System.Array.Resize(ref playgroundCache.rotation, loadParticleCount);
  5369. for (int p = particleCount; p<loadParticleCount; p++) {
  5370. transitionPosition[p] = transitionPosition[p%particleCount];
  5371. transitionColor[p].a = 0;
  5372. }
  5373. isResizing = false;
  5374. });
  5375. while (isResizing)
  5376. yield return null;
  5377. particleCache = new ParticleSystem.Particle[loadParticleCount];
  5378. shurikenParticleSystem.Emit(loadParticleCount);
  5379. shurikenParticleSystem.GetParticles(particleCache);
  5380. transitionParticleCount = loadParticleCount;
  5381. }
  5382. // Check array sizes (additions to new versions of Playground Cache after snapshot was created)
  5383. if (loadSnapshotData.lifetimeSubtraction==null || loadSnapshotData.lifetimeSubtraction.Length!=loadParticleCount)
  5384. loadSnapshotData.lifetimeSubtraction = new float[loadParticleCount];
  5385. if (loadSnapshotData.isNonBirthed==null || loadSnapshotData.isNonBirthed.Length!=loadParticleCount)
  5386. loadSnapshotData.isNonBirthed = new bool[loadParticleCount];
  5387. if (loadSnapshotData.isMasked==null || loadSnapshotData.isMasked.Length!=loadParticleCount)
  5388. loadSnapshotData.isMasked = new bool[loadParticleCount];
  5389. // Fix positioning for misplaced particles
  5390. for (int p = 0; p<loadParticleCount; p++) {
  5391. if (loadSnapshotData.position[p].y == initPosY || loadSnapshotData.life[p]>=loadParticleLifetime-loadSnapshotData.lifetimeSubtraction[p]) {
  5392. loadSnapshotData.position[p] = loadSnapshotData.targetPosition[p];
  5393. transitionColor[p].a = 0;
  5394. }
  5395. if (transitionPosition[p].y == initPosY || playgroundCache.life[p%particleCount]>=lifetime-playgroundCache.lifetimeSubtraction[p%particleCount]) {
  5396. transitionPosition[p] = loadSnapshotData.targetPosition[p];
  5397. transitionColor[p].a = 0;
  5398. }
  5399. }
  5400. // Transition
  5401. while (PlaygroundC.globalTime<transitionStartTime+((loadTransitionTime*snapshots[loadPointer].transitionMultiplier)-.001f) && inTransition && !abortTransition && loadTransition && (loadTransitionTime*snapshots[loadPointer].transitionMultiplier)>0 && currentParticleCount==particleCount) {
  5402. float currentTime = PlaygroundC.globalTime;
  5403. PlaygroundC.RunAsync (()=>{
  5404. float t = TransitionType (thisSnapshotTransition, (currentTime-transitionStartTime)/(loadTransitionTime*snapshots[loadPointer].transitionMultiplier));
  5405. for (int p = 0; p<transitionParticleCount; p++) {
  5406. if (!loadSnapshotData.isNonBirthed[p%loadParticleCount]) {
  5407. // Position
  5408. playgroundCache.position[p] = Vector3.Lerp (transitionPosition[p], liveParticles?loadSnapshotData.position[p%loadParticleCount]:loadSnapshotData.targetPosition[p%loadParticleCount], t);
  5409. if (!syncPositionsOnMainThread)
  5410. particleCache[p].position = playgroundCache.position[p];
  5411. // Color
  5412. if (!loadSnapshotData.isMasked[p%loadParticleCount]) {
  5413. playgroundCache.color[p] = Color32.Lerp (transitionColor[p], loadSnapshotData.color[p%loadParticleCount], t);
  5414. } else {
  5415. if (playgroundCache.isMasked[p])
  5416. playgroundCache.color[p] = transparent;
  5417. else playgroundCache.color[p] = Color32.Lerp (transitionColor[p], transparent, t);
  5418. }
  5419. if ((loadParticleCount<particleCount && p>=loadParticleCount))
  5420. playgroundCache.color[p].a = (byte)Mathf.Lerp (playgroundCache.color[p].a, 0f, t);
  5421. particleCache[p].color = playgroundCache.color[p];
  5422. // Size
  5423. playgroundCache.size[p] = Mathf.Lerp (transitionSize[p], loadSnapshotData.size[p%loadParticleCount], t);
  5424. particleCache[p].size = playgroundCache.size[p];
  5425. // Rotation
  5426. playgroundCache.rotation[p] = Mathf.Lerp (transitionRotation[p], loadSnapshotData.rotation[p%loadParticleCount], t);
  5427. particleCache[p].rotation = playgroundCache.rotation[p];
  5428. } else {
  5429. playgroundCache.color[p].a = (byte)Mathf.Lerp (playgroundCache.color[p].a, 0, t);
  5430. particleCache[p].color = playgroundCache.color[p];
  5431. }
  5432. }
  5433. });
  5434. yield return null;
  5435. if (firstFrameDone && currentParticleCount==particleCount) {
  5436. if (syncPositionsOnMainThread)
  5437. for (int p = 0; p<transitionParticleCount; p++)
  5438. particleCache[p].position = playgroundCache.position[p];
  5439. shurikenParticleSystem.SetParticles(particleCache, particleCache.Length);
  5440. }
  5441. firstFrameDone = true;
  5442. }
  5443. if (loadParticleCount!=particleCount && abortTransition) {
  5444. particleCache = new ParticleSystem.Particle[particleCount];
  5445. shurikenParticleSystem.Emit(particleCount);
  5446. shurikenParticleSystem.GetParticles(particleCache);
  5447. shurikenParticleSystem.SetParticles(particleCache, particleCache.Length);
  5448. System.Array.Resize(ref playgroundCache.position, particleCount);
  5449. System.Array.Resize(ref playgroundCache.color, particleCount);
  5450. System.Array.Resize(ref playgroundCache.size, particleCount);
  5451. System.Array.Resize(ref playgroundCache.rotation, particleCount);
  5452. }
  5453. lastTimeUpdated = PlaygroundC.globalTime;
  5454. cameFromNonCalculatedFrame = false;
  5455. cameFromNonEmissionFrame = false;
  5456. if (!abortTransition)
  5457. inTransition = false;
  5458. }
  5459. float TransitionType (INDIVIDUALTRANSITIONTYPEC thisTransitionType, float t) {
  5460. if (thisTransitionType==INDIVIDUALTRANSITIONTYPEC.Linear)
  5461. return t;
  5462. else if (thisTransitionType==INDIVIDUALTRANSITIONTYPEC.EaseIn)
  5463. return Mathf.Lerp (0f, 1f, 1f-Mathf.Cos(t*Mathf.PI*.5f));
  5464. else if (thisTransitionType==INDIVIDUALTRANSITIONTYPEC.EaseOut)
  5465. return Mathf.Lerp (0f, 1f, Mathf.Sin(t*Mathf.PI*.5f));
  5466. return t;
  5467. }
  5468. // Check all needed references
  5469. void CheckReferences () {
  5470. if (PlaygroundC.reference==null)
  5471. PlaygroundC.reference = FindObjectOfType<PlaygroundC>();
  5472. if (PlaygroundC.reference==null) {
  5473. PlaygroundC.reference = PlaygroundC.ResourceInstantiate("Playground Manager").GetComponent<PlaygroundC>();
  5474. }
  5475. if (playgroundCache==null)
  5476. playgroundCache = new PlaygroundCache();
  5477. if (thisInstance==null)
  5478. thisInstance = this;
  5479. if (particleSystemGameObject==null) {
  5480. particleSystemGameObject = gameObject;
  5481. particleSystemTransform = transform;
  5482. particleSystemRenderer = GetComponent<Renderer>();
  5483. shurikenParticleSystem = particleSystemGameObject.GetComponent<ParticleSystem>();
  5484. particleSystemRenderer2 = gameObject.GetComponent<ParticleSystem>().GetComponent<Renderer>() as ParticleSystemRenderer;
  5485. }
  5486. }
  5487. // YieldedRefresh makes sure that Playground Manager and simulation time is ready before this particle system
  5488. bool isYieldRefreshing = false;
  5489. bool initialized = false;
  5490. public IEnumerator YieldedRefresh () {
  5491. if (isSnapshot) yield break;
  5492. if (isYieldRefreshing) yield break;
  5493. bool okToLoadFromStart = true;
  5494. if (isSettingParticleCount || isSettingLifetime) okToLoadFromStart = false;
  5495. while (!PlaygroundC.IsReady()) yield return null;
  5496. while (isSettingParticleCount) yield return null;
  5497. while (isSettingLifetime) yield return null;
  5498. if (!prewarm && (sorting==SORTINGC.NearestNeighbor || sorting==SORTINGC.NearestNeighborReversed)) {
  5499. shurikenParticleSystem.Clear();
  5500. if (source==SOURCEC.SkinnedWorldObject) {yield return null;}
  5501. SetLifetime(thisInstance, SORTINGC.Burst, lifetime);
  5502. while (isSettingLifetime) yield return null;
  5503. isYieldRefreshing = false;
  5504. StartCoroutine(Prewarm(loop?prewarmTime:0f, 0));
  5505. yield return null;
  5506. while (isPrewarming) yield return null;
  5507. SetParticleTimeNow(thisInstance);
  5508. initialized = true;
  5509. yield break;
  5510. }
  5511. while (isSettingParticleCount || isSettingLifetime) yield return null;
  5512. isYieldRefreshing = true;
  5513. // Snapshot load
  5514. #if UNITY_EDITOR
  5515. if (!UnityEditor.EditorApplication.isPlaying)
  5516. okToLoadFromStart = false;
  5517. #endif
  5518. if (okToLoadFromStart && loadFromStart && snapshots.Count>0) {
  5519. SetLifetime(thisInstance, sorting, lifetime);
  5520. Load(loadFrom);
  5521. yield return null;
  5522. isYieldRefreshing = false;
  5523. initialized = true;
  5524. yield break;
  5525. }
  5526. SetLifetime(thisInstance, sorting, lifetime);
  5527. while (isSettingLifetime) yield return null;
  5528. yield return null;
  5529. transitionAvailable = true;
  5530. hasActiveParticles = true;
  5531. threadHadNoActiveParticles = false;
  5532. isYieldRefreshing = false;
  5533. emissionStopped = 0;
  5534. if (prewarm) {
  5535. prewarmCycles = Mathf.Clamp (prewarmCycles, 0, 1000);
  5536. StartCoroutine(Prewarm(prewarmTime, prewarmCycles));
  5537. }
  5538. while (isPrewarming) yield return null;
  5539. initialized = true;
  5540. }
  5541. /// <summary>
  5542. /// Prewarms this particle system upon boot when prewarm is set to true.
  5543. /// The prewarmTime determines where in the lifetime cycle the system should start after prewarming.
  5544. /// The amount of prewarmCycles will determine the result resolution.
  5545. /// </summary>
  5546. IEnumerator Prewarm (float time, int resolution) {
  5547. if (isPrewarming || isLoading) yield break;
  5548. if (source==SOURCEC.SkinnedWorldObject || HasTurbulence()) yield return null;
  5549. isPrewarming = true;
  5550. while (isSettingParticleCount || isSettingLifetime) yield return null;
  5551. resolution = Mathf.Clamp (resolution, 0, 1000);
  5552. int currentCycles = 0;
  5553. float storedLocalTime = 0f;
  5554. bool isPrewarmingAsync = false;
  5555. isReadyForThreadedCalculations = PrepareThreadedCalculations();
  5556. if (isReadyForThreadedCalculations) {
  5557. localTime = PlaygroundC.globalTime;
  5558. lastTimeUpdated = PlaygroundC.globalTime;
  5559. if (sorting==SORTINGC.NearestNeighbor||sorting==SORTINGC.NearestNeighborReversed) {
  5560. if (nearestNeighborOriginMethod==NEARESTNEIGHBORORIGINMETHOD.Transform && nearestNeighborOriginTransform!=null) {
  5561. nearestNeighborOriginTransformPosition = shurikenParticleSystem.simulationSpace==ParticleSystemSimulationSpace.World?
  5562. nearestNeighborOriginTransform.position:
  5563. particleSystemTransform.InverseTransformPoint(nearestNeighborOriginTransform.position);
  5564. }
  5565. storedLocalTime = localTime;
  5566. SetLifetime(thisInstance, SORTINGC.Burst, lifetime);
  5567. while (isSettingLifetime) {yield return null;}
  5568. localTime = storedLocalTime;
  5569. for (int p=0;p<particleCount;p++) SetSourcePosition(p);
  5570. storedLocalTime = localTime;
  5571. SetLifetime(thisInstance, sorting, lifetime);
  5572. while (isSettingLifetime) {yield return null;}
  5573. localTime = storedLocalTime;
  5574. for (int p=0;p<particleCount;p++) SetSourcePosition(p);
  5575. }
  5576. while (isSettingLifetime) yield return null;
  5577. if (resolution>0) {
  5578. isPrewarmingAsync = true;
  5579. PlaygroundC.RunAsync(()=>{
  5580. while (currentCycles<=resolution) {
  5581. PrewarmStepper(time, resolution);
  5582. currentCycles++;
  5583. }
  5584. if (syncPositionsOnMainThread && calculateDeltaMovement) {
  5585. for (int p = 0; p<particleCount; p++)
  5586. playgroundCache.previousTargetPosition[p] = playgroundCache.targetPosition[p];
  5587. }
  5588. isPrewarmingAsync = false;
  5589. });
  5590. }
  5591. }
  5592. while (isPrewarmingAsync)
  5593. yield return null;
  5594. isPrewarming = false;
  5595. }
  5596. void PrewarmStepper (float time, int resolution) {
  5597. localDeltaTime = ((lifetime*time)/(resolution*1f));
  5598. t = localDeltaTime;
  5599. localTime += t;
  5600. ThreadedCalculations(thisInstance);
  5601. if (HasTurbulence() && PlaygroundC.reference.turbulenceThreadMethod==ThreadMethodComponent.OneForAll)
  5602. for (int p = 0; p<particleCount; p++)
  5603. Turbulence(thisInstance, turbulenceSimplex, p, t, turbulenceType, turbulenceTimeScale, turbulenceScale, turbulenceStrength, turbulenceApplyLifetimeStrength, turbulenceLifetimeStrength);
  5604. }
  5605. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  5606. // MonoBehaviours
  5607. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  5608. void OnEnable () {
  5609. if (isSnapshot || isLoading) return;
  5610. thisInstance = this;
  5611. thisLayer = particleSystemGameObject.layer;
  5612. initialized = false;
  5613. isYieldRefreshing = false;
  5614. isSettingParticleCount = false;
  5615. isSettingLifetime = false;
  5616. hasActiveParticles = true;
  5617. threadHadNoActiveParticles = false;
  5618. isDoneThread = true;
  5619. calculate = true;
  5620. lastTimeUpdated = PlaygroundC.globalTime;
  5621. localTime = lastTimeUpdated;
  5622. emissionStopped = 0;
  5623. queueEmissionHalt = false;
  5624. hasEmitted = false;
  5625. // Set new randoms
  5626. RefreshSystemRandom();
  5627. // Set 0 size to avoid one-frame flash
  5628. shurikenParticleSystem.startSize = 0f;
  5629. if (shurikenParticleSystem.isPaused || shurikenParticleSystem.isStopped)
  5630. shurikenParticleSystem.Play();
  5631. // Set initial values
  5632. previousLifetimeValueMethod = lifetimeValueMethod;
  5633. previousLifetime = lifetime;
  5634. previousLifetimeMin = lifetimeMin;
  5635. previousEmission = emit;
  5636. previousLoop = loop;
  5637. previousNearestNeighborOrigin = nearestNeighborOrigin;
  5638. previousNearestNeighborOriginMethod = nearestNeighborOriginMethod;
  5639. previousNearestNeighborOriginVector3 = nearestNeighborOriginVector3;
  5640. previousNearestNeighborOriginTransform = nearestNeighborOriginTransform;
  5641. previousSorting = sorting;
  5642. previousParticleCount = particleCount;
  5643. previousLifetimeEmission = lifetimeEmission;
  5644. prevSource = source;
  5645. loopExceeded = false;
  5646. loopExceededOnParticle = -1;
  5647. stRot = Quaternion.identity;
  5648. stDir = new Vector3();
  5649. // Initiate all arrays by setting particle count
  5650. if (particleCache==null) {
  5651. SetParticleCount(thisInstance, particleCount);
  5652. } else {
  5653. // Clean up particle positions
  5654. SetInitialTargetPosition(thisInstance, PlaygroundC.initialTargetPosition, true);
  5655. // Clean up collision cache
  5656. hasCollisionCache = false;
  5657. collisionCache = null;
  5658. // Refresh
  5659. StartCoroutine(YieldedRefresh());
  5660. }
  5661. }
  5662. public void Start () {
  5663. if (isSnapshot || !gameObject.activeInHierarchy || !gameObject.activeSelf) return;
  5664. // Make sure all references exists
  5665. CheckReferences();
  5666. if (thisInstance.enabled) {
  5667. // Make sure that state data is initialized
  5668. if (source==SOURCEC.State)
  5669. for (int x = 0; x<states.Count; x++)
  5670. states[x].Initialize();
  5671. // Initialize
  5672. if (worldObject!=null && worldObject.transform!=null)
  5673. worldObject.Initialize();
  5674. if (skinnedWorldObject!=null && skinnedWorldObject.transform!=null)
  5675. skinnedWorldObject.Initialize();
  5676. if (projection!=null && projection.projectionTexture)
  5677. projection.Initialize();
  5678. if (manipulators.Count>0) {
  5679. for (int i = 0; i<manipulators.Count; i++)
  5680. manipulators[i].Update();
  5681. }
  5682. }
  5683. if (PlaygroundC.reference!=null) {
  5684. #if UNITY_EDITOR
  5685. if (isSnapshot && !PlaygroundC.reference.showSnapshotsInHierarchy) {
  5686. gameObject.hideFlags = HideFlags.HideInHierarchy;
  5687. return;
  5688. }
  5689. #endif
  5690. if (isSnapshot) return;
  5691. if (!PlaygroundC.reference.particleSystems.Contains(thisInstance))
  5692. PlaygroundC.reference.particleSystems.Add(thisInstance);
  5693. if (particleSystemTransform.parent==null && PlaygroundC.reference.autoGroup)
  5694. particleSystemTransform.parent = PlaygroundC.referenceTransform;
  5695. }
  5696. // Add this particle system as user of its splines
  5697. if (splines.Count>0) {
  5698. for (int i = 0; i<splines.Count; i++)
  5699. splines[i].AddUser(particleSystemTransform);
  5700. }
  5701. // Reset event controlled by-list, this will be refreshed first calculation
  5702. eventControlledBy = new List<PlaygroundParticlesC>();
  5703. //if (gameObject.activeInHierarchy && gameObject.activeSelf)
  5704. StartCoroutine (Boot());
  5705. }
  5706. public IEnumerator Boot () {
  5707. if (particleSystemGameObject.activeInHierarchy && particleSystemGameObject.activeSelf && !isLoading && enabled) {
  5708. while (PlaygroundC.reference==null) yield return null;
  5709. // Check if particle system must be combined on threaded calls to cope with particle tracking
  5710. if (PlaygroundC.reference.HasEnabledGlobalManipulators()) {
  5711. for (int m = 0; m<PlaygroundC.reference.manipulators.Count; m++) {
  5712. if (PlaygroundC.reference.manipulators[m].trackParticles && (PlaygroundC.reference.manipulators[m].affects.value & 1<<GetLayer())!=0 && PlaygroundC.reference.threadMethod!=ThreadMethod.NoThreads)
  5713. threadMethod = ThreadMethodLocal.OneForAll;
  5714. }
  5715. }
  5716. // Update time
  5717. localTime = PlaygroundC.globalTime;
  5718. lastTimeUpdated = localTime;
  5719. // Update id if needed
  5720. for (int i = 0; i<PlaygroundC.reference.particleSystems.Count; i++) {
  5721. if (PlaygroundC.reference.particleSystems[i].particleSystemId==particleSystemId && PlaygroundC.reference.particleSystems[i]!=this) {
  5722. particleSystemId = PlaygroundC.reference.particleSystems.Count-1;
  5723. break;
  5724. }
  5725. }
  5726. // Refresh
  5727. if (!isYieldRefreshing)
  5728. StartCoroutine(YieldedRefresh());
  5729. }
  5730. }
  5731. #if UNITY_EDITOR
  5732. void OnDrawGizmos () {
  5733. if (pauseCalculationWhenInvisible && calculationTriggerSizeGizmo)
  5734. Gizmos.DrawWireCube(calculationTriggerTransform==null?particleSystemTransform.position+calculationTriggerOffset:calculationTriggerTransform.position+calculationTriggerOffset, calculationTriggerSize);
  5735. }
  5736. #endif
  5737. void OnDestroy () {
  5738. // Remove any events listed by other particle systems
  5739. for (int i = 0; i<events.Count; i++) {
  5740. if (events[i].target!=null)
  5741. events[i].target.eventControlledBy.Remove (thisInstance);
  5742. }
  5743. // Remove this particle system as user of its splines
  5744. if (splines.Count>0) {
  5745. for (int i = 0; i<splines.Count; i++)
  5746. if (splines[i]!=null)
  5747. splines[i].RemoveUser(particleSystemTransform);
  5748. }
  5749. // Remove this PlaygroundParticlesC object from Particle Systems list
  5750. if (PlaygroundC.reference)
  5751. PlaygroundC.reference.particleSystems.Remove(thisInstance);
  5752. }
  5753. void OnDisable () {
  5754. initialized = false;
  5755. }
  5756. #if UNITY_EDITOR
  5757. // Select the particle system in Editor from MonoBehavior as we need one frame to initialize
  5758. public void EditorYieldSelect () {StartCoroutine (YieldSelect ());}
  5759. public IEnumerator YieldSelect () {
  5760. if (isSnapshot || Application.isPlaying) yield break;
  5761. yield return null;
  5762. if (!UnityEditor.EditorApplication.isPlaying)
  5763. UnityEditor.Selection.activeGameObject = particleSystemGameObject;
  5764. }
  5765. #endif
  5766. }
  5767. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  5768. // MeshParticles - Extension class for PlaygroundParticlesC which creates mesh states.
  5769. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  5770. public class MeshParticles : PlaygroundParticlesC {
  5771. public static PlaygroundParticlesC CreateMeshParticles (Mesh[] meshes, Texture2D[] textures, Texture2D[] heightMap, string name, Vector3 position, Quaternion rotation, float particleScale, Vector3[] offsets, Material material) {
  5772. PlaygroundParticlesC meshParticles = PlaygroundParticlesC.CreateParticleObject(name,position,rotation,particleScale,material);
  5773. meshParticles.states = new List<ParticleStateC>();
  5774. int[] quantityList = new int[meshes.Length];
  5775. int i = 0;
  5776. for (; i<textures.Length; i++)
  5777. quantityList[i] = meshes[i].vertexCount;
  5778. meshParticles.particleCache = new ParticleSystem.Particle[quantityList[PlaygroundC.Largest(quantityList)]];
  5779. meshParticles.shurikenParticleSystem.Emit(meshParticles.particleCache.Length);
  5780. meshParticles.shurikenParticleSystem.GetParticles(meshParticles.particleCache);
  5781. for (i = 0; i<meshes.Length; i++) {
  5782. meshParticles.states.Add(new ParticleStateC());
  5783. meshParticles.states[0].ConstructParticles(meshes[i],textures[i],particleScale,offsets[i], "State "+i,null);
  5784. }
  5785. meshParticles.UpdateSystem();
  5786. PlaygroundC.particlesQuantity++;
  5787. return meshParticles;
  5788. }
  5789. public static void Add (PlaygroundParticlesC meshParticles, Mesh mesh, float scale, Vector3 offset, string stateName, Transform stateTransform) {
  5790. meshParticles.states.Add(new ParticleStateC());
  5791. meshParticles.states[meshParticles.states.Count-1].ConstructParticles(mesh,scale,offset,stateName,stateTransform);
  5792. }
  5793. public static void Add (PlaygroundParticlesC meshParticles, Mesh mesh, Texture2D texture, float scale, Vector3 offset, string stateName, Transform stateTransform) {
  5794. meshParticles.states.Add(new ParticleStateC());
  5795. meshParticles.states[meshParticles.states.Count-1].ConstructParticles(mesh,texture,scale,offset,stateName,stateTransform);
  5796. }
  5797. }
  5798. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  5799. // Cache
  5800. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  5801. /// <summary>
  5802. /// Holds information for a particle system Snapshot.
  5803. /// </summary>
  5804. [Serializable]
  5805. public class PlaygroundSave {
  5806. /// <summary>
  5807. /// The name of this PlaygroundSave.
  5808. /// </summary>
  5809. [HideInInspector] public string name;
  5810. /// <summary>
  5811. /// The global time when this PlaygroundSave was created.
  5812. /// </summary>
  5813. [HideInInspector] public float time;
  5814. /// <summary>
  5815. /// The lifetime of the particle system.
  5816. /// </summary>
  5817. [HideInInspector] public float lifetime;
  5818. /// <summary>
  5819. /// The particle count of the particle system.
  5820. /// </summary>
  5821. [HideInInspector] public int particleCount;
  5822. /// <summary>
  5823. /// The version this snapshot was made with.
  5824. /// </summary>
  5825. [HideInInspector] public float version = 0;
  5826. /// <summary>
  5827. /// The cached settings of the particle system (Particles are stored in settings.snapshotData).
  5828. /// </summary>
  5829. [HideInInspector] public PlaygroundParticlesC settings;
  5830. /// <summary>
  5831. /// The stored transform information.
  5832. /// </summary>
  5833. [HideInInspector] public PlaygroundTransformC transform;
  5834. /// <summary>
  5835. /// The stored material.
  5836. /// </summary>
  5837. [HideInInspector] public Material material;
  5838. /// <summary>
  5839. /// The method to load data from this PlaygroundSave.
  5840. /// 0 = Settings+Particles, 1 = Settings only, 2 = Particles only
  5841. /// </summary>
  5842. [HideInInspector] public int loadMode = 0;
  5843. /// <summary>
  5844. /// Should the stored transform information load?
  5845. /// </summary>
  5846. [HideInInspector] public bool loadTransform = true;
  5847. /// <summary>
  5848. /// Should the stored material load?
  5849. /// </summary>
  5850. [HideInInspector] public bool loadMaterial = true;
  5851. /// <summary>
  5852. /// Should the material load before or after transition?
  5853. /// </summary>
  5854. [HideInInspector] public bool setMaterialAfterTransition = true;
  5855. /// <summary>
  5856. /// The multiplier of transition time.
  5857. /// </summary>
  5858. [HideInInspector] public float transitionMultiplier = 1f;
  5859. /// <summary>
  5860. /// The transition type of this PlaygroundSave.
  5861. /// </summary>
  5862. [HideInInspector] public INDIVIDUALTRANSITIONTYPEC transitionType;
  5863. [HideInInspector] public bool unfolded = false;
  5864. bool isLoading = false;
  5865. bool isSaving = false;
  5866. /// <summary>
  5867. /// Determines whether this PlaygroundSave is loading.
  5868. /// </summary>
  5869. /// <returns><c>true</c> if this PlaygroundSave is loading; otherwise, <c>false</c>.</returns>
  5870. public bool IsLoading () {
  5871. return isLoading;
  5872. }
  5873. /// <summary>
  5874. /// Determines whether this PlaygroundSave is saving.
  5875. /// </summary>
  5876. /// <returns><c>true</c> if this PlaygroundSave is saving; otherwise, <c>false</c>.</returns>
  5877. public bool IsSaving () {
  5878. return isSaving;
  5879. }
  5880. /// <summary>
  5881. /// Loads the data from this PlaygroundSave.
  5882. /// </summary>
  5883. /// <param name="loadTo">Load to.</param>
  5884. public void Load (PlaygroundParticlesC loadTo) {
  5885. if (loadMaterial && (setMaterialAfterTransition||!loadTo.loadTransition) && settings.particleSystemRenderer.sharedMaterial!=null)
  5886. loadTo.particleSystemRenderer.sharedMaterial = settings.particleSystemRenderer.sharedMaterial;
  5887. isLoading = true;
  5888. PlaygroundC.RunAsync(()=>{
  5889. switch (loadMode) {
  5890. case 0: settings.CopyTo(loadTo); loadTo.playgroundCache = settings.snapshotData.Clone(); break;
  5891. case 1: settings.CopyTo(loadTo); break;
  5892. case 2: loadTo.playgroundCache = settings.snapshotData.Clone(); break;
  5893. default: settings.CopyTo(loadTo); loadTo.playgroundCache = settings.snapshotData.Clone(); break;
  5894. }
  5895. isLoading = false;
  5896. });
  5897. if (loadTransform)
  5898. transform.GetFromTransform (loadTo.particleSystemTransform);
  5899. }
  5900. /// <summary>
  5901. /// Saves the data into this PlaygroundSave.
  5902. /// </summary>
  5903. /// <param name="playgroundParticles">Playground particles.</param>
  5904. public void Save (PlaygroundParticlesC playgroundParticles) {
  5905. isSaving = true;
  5906. transform = new PlaygroundTransformC();
  5907. transform.SetFromTransform (playgroundParticles.particleSystemTransform);
  5908. settings.particleSystemRenderer.sharedMaterial = playgroundParticles.particleSystemRenderer.sharedMaterial;
  5909. PlaygroundC.RunAsync(()=>{
  5910. playgroundParticles.CopyTo(settings);
  5911. settings.snapshotData = playgroundParticles.playgroundCache.Clone();
  5912. isSaving = false;
  5913. });
  5914. }
  5915. /// <summary>
  5916. /// Returns a copy of this PlaygroundSave.
  5917. /// </summary>
  5918. public PlaygroundSave Clone () {
  5919. PlaygroundSave playgroundSave = new PlaygroundSave();
  5920. settings.CopyTo(playgroundSave.settings);
  5921. playgroundSave.name = name;
  5922. playgroundSave.time = time;
  5923. playgroundSave.lifetime = lifetime;
  5924. playgroundSave.particleCount = particleCount;
  5925. playgroundSave.loadMode = loadMode;
  5926. playgroundSave.loadTransform = loadTransform;
  5927. playgroundSave.loadMaterial = loadMaterial;
  5928. playgroundSave.setMaterialAfterTransition = setMaterialAfterTransition;
  5929. playgroundSave.material = material;
  5930. playgroundSave.transitionMultiplier = transitionMultiplier;
  5931. playgroundSave.transitionType = transitionType;
  5932. playgroundSave.unfolded = unfolded;
  5933. return playgroundSave;
  5934. }
  5935. }
  5936. /// <summary>
  5937. /// Holds information for a PlaygroundEventParticle. The Playground Event Particle contains detailed data upon an event and is sent towards any event listeners.
  5938. /// </summary>
  5939. [Serializable]
  5940. public class PlaygroundEventParticle {
  5941. /// <summary>
  5942. /// The initial size of this particle.
  5943. /// </summary>
  5944. [HideInInspector] public float initialSize;
  5945. /// <summary>
  5946. /// The lifetime size of this particle.
  5947. /// </summary>
  5948. [HideInInspector] public float size;
  5949. /// <summary>
  5950. /// The rotation of this particle.
  5951. /// </summary>
  5952. [HideInInspector] public float rotation;
  5953. /// <summary>
  5954. /// The lifetime of this particle.
  5955. /// </summary>
  5956. [HideInInspector] public float life;
  5957. /// <summary>
  5958. /// The total time this particle will live. This will take lifetime subtraction into account.
  5959. /// </summary>
  5960. [HideInInspector] public float totalLifetime;
  5961. /// <summary>
  5962. /// The time of birth for this particle.
  5963. /// </summary>
  5964. [HideInInspector] public float birth;
  5965. /// <summary>
  5966. /// The delayed time of birth when emission has changed.
  5967. /// </summary>
  5968. [HideInInspector] public float birthDelay;
  5969. /// <summary>
  5970. /// The time of death for this particle.
  5971. /// </summary>
  5972. [HideInInspector] public float death;
  5973. /// <summary>
  5974. /// The emission for this particle (controlled by emission rate).
  5975. /// </summary>
  5976. [HideInInspector] public bool emission;
  5977. /// <summary>
  5978. /// Determines if this particle should rebirth.
  5979. /// </summary>
  5980. [HideInInspector] public bool rebirth;
  5981. /// <summary>
  5982. /// The offset in birth-death (sorting).
  5983. /// </summary>
  5984. [HideInInspector] public float lifetimeOffset;
  5985. /// <summary>
  5986. /// The velocity of this particle.
  5987. /// </summary>
  5988. [HideInInspector] public Vector3 velocity;
  5989. /// <summary>
  5990. /// The initial velocity of this particle.
  5991. /// </summary>
  5992. [HideInInspector] public Vector3 initialVelocity;
  5993. /// <summary>
  5994. /// The initial local velocity of this particle.
  5995. /// </summary>
  5996. [HideInInspector] public Vector3 initialLocalVelocity;
  5997. /// <summary>
  5998. /// The position of this particle.
  5999. /// </summary>
  6000. [HideInInspector] public Vector3 position;
  6001. /// <summary>
  6002. /// The source position at birth for this particle.
  6003. /// </summary>
  6004. [HideInInspector] public Vector3 targetPosition;
  6005. /// <summary>
  6006. /// The source direction at birth for this particle.
  6007. /// </summary>
  6008. [HideInInspector] public Vector3 targetDirection;
  6009. /// <summary>
  6010. /// The previous source position for this particle (used to calculate delta movement).
  6011. /// </summary>
  6012. [HideInInspector] public Vector3 previousTargetPosition;
  6013. /// <summary>
  6014. /// The previous calculated frame's particle position.
  6015. /// </summary>
  6016. [HideInInspector] public Vector3 previousParticlePosition;
  6017. /// <summary>
  6018. /// The calculated particle position for collision.
  6019. /// </summary>
  6020. [HideInInspector] public Vector3 collisionParticlePosition;
  6021. /// <summary>
  6022. /// The delta to compensate for moving particles in local space.
  6023. /// </summary>
  6024. [HideInInspector] public Vector3 localSpaceMovementCompensation;
  6025. /// <summary>
  6026. /// The scattered position to apply on this particle birth.
  6027. /// </summary>
  6028. [HideInInspector] public Vector3 scatterPosition;
  6029. /// <summary>
  6030. /// The initial rotation of this particle.
  6031. /// </summary>
  6032. [HideInInspector] public float initialRotation;
  6033. /// <summary>
  6034. /// The rotation speed of this particle.
  6035. /// </summary>
  6036. [HideInInspector] public float rotationSpeed;
  6037. /// <summary>
  6038. /// The current color of this particle.
  6039. /// </summary>
  6040. [HideInInspector] public Color32 color;
  6041. /// <summary>
  6042. /// The color set from script of this particle.
  6043. /// </summary>
  6044. [HideInInspector] public Color32 scriptedColor;
  6045. /// <summary>
  6046. /// The set source color.
  6047. /// </summary>
  6048. [HideInInspector] public Color32 initialColor;
  6049. /// <summary>
  6050. /// The color gradient for this particle if Color Source is set to LifetimeColors.
  6051. /// </summary>
  6052. [HideInInspector] public int lifetimeColorId;
  6053. /// <summary>
  6054. /// The particle does not respond to forces during its lifetime.
  6055. /// </summary>
  6056. [HideInInspector] public bool noForce;
  6057. /// <summary>
  6058. /// The particle is non birthed.
  6059. /// </summary>
  6060. [HideInInspector] public bool isNonBirthed;
  6061. /// <summary>
  6062. /// The particle is in its first loop.
  6063. /// </summary>
  6064. [HideInInspector] public bool isFirstLoop;
  6065. /// <summary>
  6066. /// The id of this particle.
  6067. /// </summary>
  6068. [HideInInspector] public int particleId;
  6069. /// <summary>
  6070. /// The id of the particle system this particle belongs to (list position in Playground Manager).
  6071. /// </summary>
  6072. [HideInInspector] public int particleSystemId;
  6073. /// <summary>
  6074. /// The id of the last manipulator affecting this particle.
  6075. /// </summary>
  6076. [HideInInspector] public int manipulatorId;
  6077. /// <summary>
  6078. /// The interaction with property manipulators of this particle.
  6079. /// </summary>
  6080. [HideInInspector] public bool changedByProperty;
  6081. /// <summary>
  6082. /// The interaction with property manipulators that change color of this particle.
  6083. /// </summary>
  6084. [HideInInspector] public bool changedByPropertyColor;
  6085. /// <summary>
  6086. /// The interaction with property manipulators that change color over time of this particle.
  6087. /// </summary>
  6088. [HideInInspector] public bool changedByPropertyColorLerp;
  6089. /// <summary>
  6090. /// The interaction with property manipulators that change color and wants to keep alpha.
  6091. /// </summary>
  6092. [HideInInspector] public bool changedByPropertyColorKeepAlpha;
  6093. /// <summary>
  6094. /// The interaction with property manipulators that change size of this particle.
  6095. /// </summary>
  6096. [HideInInspector] public bool changedByPropertySize;
  6097. /// <summary>
  6098. /// The interaction with property manipulators that change target of this particle.
  6099. /// </summary>
  6100. [HideInInspector] public bool changedByPropertyTarget;
  6101. /// <summary>
  6102. /// The interaction with death manipulators that forces a particle to a sooner end.
  6103. /// </summary>
  6104. [HideInInspector] public bool changedByPropertyDeath;
  6105. /// <summary>
  6106. /// The property target pointer for this particle.
  6107. /// </summary>
  6108. [HideInInspector] public int propertyTarget;
  6109. /// <summary>
  6110. /// The property target id for this particle (pairing a particle's target to a manipulator).
  6111. /// </summary>
  6112. [HideInInspector] public int propertyId;
  6113. /// <summary>
  6114. /// The property color id for this particles (pairing a particle's color to a manipulator).
  6115. /// </summary>
  6116. [HideInInspector] public int propertyColorId;
  6117. /// <summary>
  6118. /// The manipulator id to exclude to not affect this particle.
  6119. /// </summary>
  6120. [HideInInspector] public int excludeFromManipulatorId;
  6121. /// <summary>
  6122. /// Is this particle masked?
  6123. /// </summary>
  6124. [HideInInspector] public bool isMasked;
  6125. /// <summary>
  6126. /// The alpha of this masked particle.
  6127. /// </summary>
  6128. [HideInInspector] public float maskAlpha;
  6129. /// <summary>
  6130. /// The collision transform of this collided particle.
  6131. /// </summary>
  6132. [HideInInspector] public Transform collisionTransform;
  6133. /// <summary>
  6134. /// The collision collider of this collided particle (3d).
  6135. /// </summary>
  6136. [HideInInspector] public Collider collisionCollider;
  6137. /// <summary>
  6138. /// The collision collider of this collided particle (2d).
  6139. /// </summary>
  6140. [HideInInspector] public Collider2D collisionCollider2D;
  6141. /// <summary>
  6142. /// Copies this PlaygroundEventParticle.
  6143. /// </summary>
  6144. public PlaygroundEventParticle Clone () {
  6145. PlaygroundEventParticle playgroundEventParticle = new PlaygroundEventParticle();
  6146. playgroundEventParticle.initialSize = initialSize;
  6147. playgroundEventParticle.size = size;
  6148. playgroundEventParticle.life = life;
  6149. playgroundEventParticle.totalLifetime = totalLifetime;
  6150. playgroundEventParticle.rotation = rotation;
  6151. playgroundEventParticle.birth = birth;
  6152. playgroundEventParticle.birthDelay = birthDelay;
  6153. playgroundEventParticle.death = death;
  6154. playgroundEventParticle.emission = emission;
  6155. playgroundEventParticle.rebirth = rebirth;
  6156. playgroundEventParticle.lifetimeOffset = lifetimeOffset;
  6157. playgroundEventParticle.velocity = velocity;
  6158. playgroundEventParticle.initialVelocity = initialVelocity;
  6159. playgroundEventParticle.initialLocalVelocity = initialLocalVelocity;
  6160. playgroundEventParticle.position = position;
  6161. playgroundEventParticle.targetPosition = targetPosition;
  6162. playgroundEventParticle.targetDirection = targetDirection;
  6163. playgroundEventParticle.previousTargetPosition = previousTargetPosition;
  6164. playgroundEventParticle.previousParticlePosition = previousParticlePosition;
  6165. playgroundEventParticle.collisionParticlePosition = collisionParticlePosition;
  6166. playgroundEventParticle.localSpaceMovementCompensation = localSpaceMovementCompensation;
  6167. playgroundEventParticle.scatterPosition = scatterPosition;
  6168. playgroundEventParticle.initialRotation = initialRotation;
  6169. playgroundEventParticle.rotationSpeed = rotationSpeed;
  6170. playgroundEventParticle.color = color;
  6171. playgroundEventParticle.scriptedColor = scriptedColor;
  6172. playgroundEventParticle.initialColor = initialColor;
  6173. playgroundEventParticle.lifetimeColorId = lifetimeColorId;
  6174. playgroundEventParticle.noForce = noForce;
  6175. playgroundEventParticle.changedByProperty = changedByProperty;
  6176. playgroundEventParticle.changedByPropertyColor = changedByPropertyColor;
  6177. playgroundEventParticle.changedByPropertyColorLerp = changedByPropertyColorLerp;
  6178. playgroundEventParticle.changedByPropertyColorKeepAlpha = changedByPropertyColorKeepAlpha;
  6179. playgroundEventParticle.changedByPropertySize = changedByPropertySize;
  6180. playgroundEventParticle.changedByPropertyTarget = changedByPropertyTarget;
  6181. playgroundEventParticle.changedByPropertyDeath = changedByPropertyDeath;
  6182. playgroundEventParticle.propertyTarget = propertyTarget;
  6183. playgroundEventParticle.propertyId = propertyId;
  6184. playgroundEventParticle.excludeFromManipulatorId = excludeFromManipulatorId;
  6185. playgroundEventParticle.propertyColorId = propertyColorId;
  6186. playgroundEventParticle.particleId = particleId;
  6187. playgroundEventParticle.particleSystemId = particleSystemId;
  6188. playgroundEventParticle.manipulatorId = manipulatorId;
  6189. playgroundEventParticle.isMasked = isMasked;
  6190. playgroundEventParticle.maskAlpha = maskAlpha;
  6191. playgroundEventParticle.collisionTransform = collisionTransform;
  6192. playgroundEventParticle.collisionCollider = collisionCollider;
  6193. playgroundEventParticle.collisionCollider2D = collisionCollider2D;
  6194. return playgroundEventParticle;
  6195. }
  6196. }
  6197. /// <summary>
  6198. /// The Playground Cache contains all data for particles in built-in arrays.
  6199. /// </summary>
  6200. [Serializable]
  6201. public class PlaygroundCache {
  6202. /// <summary>
  6203. /// The initial size of each particle.
  6204. /// </summary>
  6205. [HideInInspector] public float[] initialSize;
  6206. /// <summary>
  6207. /// The lifetime size of each particle.
  6208. /// </summary>
  6209. [HideInInspector] public float[] size;
  6210. /// <summary>
  6211. /// The rotation of each particle.
  6212. /// </summary>
  6213. [HideInInspector] public float[] rotation;
  6214. /// <summary>
  6215. /// The current lifetime of each particle.
  6216. /// </summary>
  6217. [HideInInspector] public float[] life;
  6218. /// <summary>
  6219. /// The lifetime subtraction of each particle (applied when using random between two values).
  6220. /// </summary>
  6221. [HideInInspector] public float[] lifetimeSubtraction;
  6222. /// <summary>
  6223. /// The time of birth for each particle.
  6224. /// </summary>
  6225. [HideInInspector] public float[] birth;
  6226. /// <summary>
  6227. /// The delayed time of birth when emission has changed.
  6228. /// </summary>
  6229. [HideInInspector] public float[] birthDelay;
  6230. /// <summary>
  6231. /// The time of death for each particle.
  6232. /// </summary>
  6233. [HideInInspector] public float[] death;
  6234. /// <summary>
  6235. /// The emission for each particle (controlled by emission rate).
  6236. /// </summary>
  6237. [HideInInspector] public bool[] emission;
  6238. /// <summary>
  6239. /// The rebirth for each particle.
  6240. /// </summary>
  6241. [HideInInspector] public bool[] rebirth;
  6242. /// <summary>
  6243. /// The offset in birth-death (sorting).
  6244. /// </summary>
  6245. [HideInInspector] public float[] lifetimeOffset;
  6246. /// <summary>
  6247. /// The velocity of each particle in this PlaygroundParticles.
  6248. /// </summary>
  6249. [HideInInspector] public Vector3[] velocity;
  6250. /// <summary>
  6251. /// The initial velocity of each particle in this PlaygroundParticles.
  6252. /// </summary>
  6253. [HideInInspector] public Vector3[] initialVelocity;
  6254. /// <summary>
  6255. /// The initial local velocity of each particle in this PlaygroundParticles.
  6256. /// </summary>
  6257. [HideInInspector] public Vector3[] initialLocalVelocity;
  6258. /// <summary>
  6259. /// The position of each particle in this PlaygroundParticles.
  6260. /// </summary>
  6261. [HideInInspector] public Vector3[] position;
  6262. /// <summary>
  6263. /// The source position for each particle.
  6264. /// </summary>
  6265. [HideInInspector] public Vector3[] targetPosition;
  6266. /// <summary>
  6267. /// The source direction for each particle.
  6268. /// </summary>
  6269. [HideInInspector] public Vector3[] targetDirection;
  6270. /// <summary>
  6271. /// The previous source position for each particle (used to calculate delta movement).
  6272. /// </summary>
  6273. [HideInInspector] public Vector3[] previousTargetPosition;
  6274. /// <summary>
  6275. /// The previous calculated frame's particle position.
  6276. /// </summary>
  6277. [HideInInspector] public Vector3[] previousParticlePosition;
  6278. /// <summary>
  6279. /// The calculated particle position for collision (depending on collision stepper).
  6280. /// </summary>
  6281. [HideInInspector] public Vector3[] collisionParticlePosition;
  6282. /// <summary>
  6283. /// The delta to compensate for moving particles in local space.
  6284. /// </summary>
  6285. [HideInInspector] public Vector3[] localSpaceMovementCompensation;
  6286. /// <summary>
  6287. /// The scattered position to apply on each particle birth in this PlaygroundParticles.
  6288. /// </summary>
  6289. [HideInInspector] public Vector3[] scatterPosition;
  6290. /// <summary>
  6291. /// The initial rotation of each particle in this PlaygroundParticles.
  6292. /// </summary>
  6293. [HideInInspector] public float[] initialRotation;
  6294. /// <summary>
  6295. /// The rotation speed of each particle in this PlaygroundParticles.
  6296. /// </summary>
  6297. [HideInInspector] public float[] rotationSpeed;
  6298. /// <summary>
  6299. /// The color of each particle in this PlaygroundParticles.
  6300. /// </summary>
  6301. [HideInInspector] public Color32[] color;
  6302. /// <summary>
  6303. /// The color set from script of each particle in this PlaygroundParticles.
  6304. /// </summary>
  6305. [HideInInspector] public Color32[] scriptedColor;
  6306. /// <summary>
  6307. /// The set source color.
  6308. /// </summary>
  6309. [HideInInspector] public Color32[] initialColor;
  6310. /// <summary>
  6311. /// The color gradient for each particle if Color Source is set to LifetimeColors.
  6312. /// </summary>
  6313. [HideInInspector] public int[] lifetimeColorId;
  6314. /// <summary>
  6315. /// The particle will no longer respond to any forces during its lifetime.
  6316. /// </summary>
  6317. [HideInInspector] public bool[] noForce;
  6318. /// <summary>
  6319. /// The particle is not birthed yet.
  6320. /// </summary>
  6321. [HideInInspector] public bool[] isNonBirthed;
  6322. /// <summary>
  6323. /// The particle is in its first loop.
  6324. /// </summary>
  6325. [HideInInspector] public bool[] isFirstLoop;
  6326. /// <summary>
  6327. /// The particle is set to simulate.
  6328. /// </summary>
  6329. [HideInInspector] public bool[] simulate;
  6330. /// <summary>
  6331. /// Determines if the particle is calculated during this frame.
  6332. /// This helps to see if a particle has gotten its expected values during this frame or if it will be ready later.
  6333. /// The method is used when syncing particles onto the Main-Thread to ensure smooth movement upon heavier calculated particle systems.
  6334. /// </summary>
  6335. [HideInInspector] public bool[] isCalculatedThisFrame;
  6336. // Manipulator specific arrays:
  6337. /// <summary>
  6338. /// The interaction with property manipulators of each particle.
  6339. /// </summary>
  6340. [HideInInspector] public bool[] changedByProperty;
  6341. /// <summary>
  6342. /// The interaction with property manipulators that change color of each particle.
  6343. /// </summary>
  6344. [HideInInspector] public bool[] changedByPropertyColor;
  6345. /// <summary>
  6346. /// The interaction with property manipulators that change color over time of each particle.
  6347. /// </summary>
  6348. [HideInInspector] public bool[] changedByPropertyColorLerp;
  6349. /// <summary>
  6350. /// The interaction with property manipulators that change color and wants to keep alpha.
  6351. /// </summary>
  6352. [HideInInspector] public bool[] changedByPropertyColorKeepAlpha;
  6353. /// <summary>
  6354. /// The interaction with property manipulators that change size of each particle.
  6355. /// </summary>
  6356. [HideInInspector] public bool[] changedByPropertySize;
  6357. /// <summary>
  6358. /// The interaction with property manipulators that change target of each particle.
  6359. /// </summary>
  6360. [HideInInspector] public bool[] changedByPropertyTarget;
  6361. /// <summary>
  6362. /// The interaction with death manipulators that forces a particle to a sooner end.
  6363. /// </summary>
  6364. [HideInInspector] public bool[] changedByPropertyDeath;
  6365. /// <summary>
  6366. /// The property target pointer for each particle.
  6367. /// </summary>
  6368. [HideInInspector] public int[] propertyTarget;
  6369. /// <summary>
  6370. /// The property target id for each particle (pairing a particle's target to a manipulator).
  6371. /// </summary>
  6372. [HideInInspector] public int[] propertyId;
  6373. /// <summary>
  6374. /// The property color id for each particles (pairing a particle's color to a manipulator.
  6375. /// </summary>
  6376. [HideInInspector] public int[] propertyColorId;
  6377. /// <summary>
  6378. /// The id of the last manipulator affecting this particle.
  6379. /// </summary>
  6380. [HideInInspector] public int[] manipulatorId;
  6381. /// <summary>
  6382. /// The id of manipulator to not affect this particle.
  6383. /// </summary>
  6384. [HideInInspector] public int[] excludeFromManipulatorId;
  6385. /// <summary>
  6386. /// Is this particle masked?
  6387. /// </summary>
  6388. [HideInInspector] public bool[] isMasked;
  6389. /// <summary>
  6390. /// The alpha of this masked particle.
  6391. /// </summary>
  6392. [HideInInspector] public float[] maskAlpha;
  6393. /// <summary>
  6394. /// The sorting of the masked particles.
  6395. /// </summary>
  6396. [HideInInspector] public int[] maskSorting;
  6397. /// <summary>
  6398. /// Copies this PlaygroundCache.
  6399. /// </summary>
  6400. public PlaygroundCache Clone () {
  6401. PlaygroundCache playgroundCache = new PlaygroundCache();
  6402. playgroundCache.initialSize = initialSize.Clone() as float[];
  6403. playgroundCache.size = size.Clone () as float[];
  6404. playgroundCache.life = life.Clone() as float[];
  6405. playgroundCache.lifetimeSubtraction = lifetimeSubtraction.Clone() as float[];
  6406. playgroundCache.rotation = rotation.Clone() as float[];
  6407. playgroundCache.birth = birth.Clone() as float[];
  6408. playgroundCache.birthDelay = birthDelay.Clone() as float[];
  6409. playgroundCache.death = death.Clone() as float[];
  6410. playgroundCache.emission = emission.Clone() as bool[];
  6411. playgroundCache.rebirth = rebirth.Clone() as bool[];
  6412. playgroundCache.lifetimeOffset = lifetimeOffset.Clone() as float[];
  6413. playgroundCache.velocity = velocity.Clone() as Vector3[];
  6414. playgroundCache.initialVelocity = initialVelocity.Clone() as Vector3[];
  6415. playgroundCache.initialLocalVelocity = initialLocalVelocity.Clone() as Vector3[];
  6416. playgroundCache.position = position.Clone () as Vector3[];
  6417. playgroundCache.targetPosition = targetPosition.Clone() as Vector3[];
  6418. playgroundCache.targetDirection = targetDirection.Clone() as Vector3[];
  6419. playgroundCache.previousTargetPosition = previousTargetPosition.Clone() as Vector3[];
  6420. playgroundCache.previousParticlePosition = previousParticlePosition.Clone() as Vector3[];
  6421. playgroundCache.collisionParticlePosition = collisionParticlePosition.Clone () as Vector3[];
  6422. playgroundCache.localSpaceMovementCompensation = localSpaceMovementCompensation.Clone() as Vector3[];
  6423. playgroundCache.scatterPosition = scatterPosition.Clone() as Vector3[];
  6424. playgroundCache.initialRotation = initialRotation.Clone() as float[];
  6425. playgroundCache.rotationSpeed = rotationSpeed.Clone() as float[];
  6426. playgroundCache.color = color.Clone() as Color32[];
  6427. playgroundCache.scriptedColor = scriptedColor.Clone() as Color32[];
  6428. playgroundCache.initialColor = initialColor.Clone () as Color32[];
  6429. playgroundCache.lifetimeColorId = lifetimeColorId.Clone () as int[];
  6430. playgroundCache.noForce = noForce.Clone() as bool[];
  6431. playgroundCache.changedByProperty = changedByProperty.Clone() as bool[];
  6432. playgroundCache.changedByPropertyColor = changedByPropertyColor.Clone() as bool[];
  6433. playgroundCache.changedByPropertyColorLerp = changedByPropertyColorLerp.Clone() as bool[];
  6434. playgroundCache.changedByPropertyColorKeepAlpha = changedByPropertyColorKeepAlpha.Clone () as bool[];
  6435. playgroundCache.changedByPropertySize = changedByPropertySize.Clone() as bool[];
  6436. playgroundCache.changedByPropertyTarget = changedByPropertyTarget.Clone() as bool[];
  6437. playgroundCache.changedByPropertyDeath = changedByPropertyDeath.Clone() as bool[];
  6438. playgroundCache.propertyTarget = propertyTarget.Clone() as int[];
  6439. playgroundCache.propertyId = propertyId.Clone() as int[];
  6440. playgroundCache.excludeFromManipulatorId = excludeFromManipulatorId.Clone() as int[];
  6441. playgroundCache.propertyColorId = propertyColorId.Clone() as int[];
  6442. playgroundCache.manipulatorId = manipulatorId.Clone() as int[];
  6443. playgroundCache.isMasked = isMasked.Clone() as bool[];
  6444. playgroundCache.maskAlpha = maskAlpha.Clone() as float[];
  6445. if (maskSorting!=null) playgroundCache.maskSorting = maskSorting.Clone() as int[];
  6446. playgroundCache.isNonBirthed = isNonBirthed.Clone() as bool[];
  6447. playgroundCache.isFirstLoop = isFirstLoop.Clone() as bool[];
  6448. playgroundCache.simulate = simulate.Clone() as bool[];
  6449. playgroundCache.isCalculatedThisFrame = isCalculatedThisFrame.Clone() as bool[];
  6450. return playgroundCache;
  6451. }
  6452. }
  6453. /// <summary>
  6454. /// The collision cache contains information of all particle collisions.
  6455. /// </summary>
  6456. [Serializable]
  6457. public class CollisionCache {
  6458. /// <summary>
  6459. /// Determines if a particle has collided with any colliders during its lifetime.
  6460. /// </summary>
  6461. [HideInInspector] public bool[] hasCollided;
  6462. /// <summary>
  6463. /// The position in world space a particle collided last. This will not update further during a particle's lifetime if using "sticky" particles.
  6464. /// </summary>
  6465. [HideInInspector] public Vector3[] collisionPosition;
  6466. /// <summary>
  6467. /// The collision normal of the surface the particle collided last. This will not update further during a particle's lifetime if using "sticky" particles.
  6468. /// </summary>
  6469. [HideInInspector] public Vector3[] collisionNormal;
  6470. /// <summary>
  6471. /// The transform of the object a particle collided with last. This will not update further during a particle's lifetime if using "sticky" particles.
  6472. /// </summary>
  6473. [HideInInspector] public Transform[] collisionTransform;
  6474. [HideInInspector] public GameObject[] collisionGameObject;
  6475. /// <summary>
  6476. /// The position seen by the collision object's transform based on InverseTransformPoint.
  6477. /// </summary>
  6478. [HideInInspector] public Vector3[] collisionTransformPosition;
  6479. /// <summary>
  6480. /// The sticky position of a particle. This contains the position for particles that stick to their collided surface.
  6481. /// </summary>
  6482. [HideInInspector] public Vector3[] stickyPosition;
  6483. public CollisionCache (int amount) {
  6484. hasCollided = new bool[amount];
  6485. collisionPosition = new Vector3[amount];
  6486. collisionNormal = new Vector3[amount];
  6487. collisionTransform = new Transform[amount];
  6488. collisionGameObject = new GameObject[amount];
  6489. collisionTransformPosition = new Vector3[amount];
  6490. stickyPosition = new Vector3[amount];
  6491. }
  6492. /// <summary>
  6493. /// Sets the specified collision position, normal and transform at index.
  6494. /// </summary>
  6495. /// <param name="index">Index.</param>
  6496. /// <param name="position">Position.</param>
  6497. /// <param name="normal">Normal.</param>
  6498. /// <param name="transform">Transform.</param>
  6499. public void Set (int index, Vector3 position, Vector3 normal, Transform transform) {
  6500. index = index%hasCollided.Length;
  6501. hasCollided[index] = true;
  6502. collisionPosition[index] = position;
  6503. collisionNormal[index] = normal;
  6504. collisionTransform[index] = transform;
  6505. collisionGameObject[index] = transform.gameObject;
  6506. }
  6507. public void SetSticky (int index, Vector3 position, Vector3 normal, float offset, Transform transform) {
  6508. index = index%hasCollided.Length;
  6509. hasCollided[index] = true;
  6510. collisionPosition[index] = position;
  6511. collisionNormal[index] = normal;
  6512. collisionTransform[index] = transform;
  6513. collisionGameObject[index] = transform.gameObject;
  6514. collisionTransformPosition[index] = transform.InverseTransformPoint(position+(collisionNormal[index]*offset));
  6515. }
  6516. /// <summary>
  6517. /// UpdateStickyPosition sets the sticky position based on the collision transform for a particle. The sticky position will update automatically when using "sticky" particles.
  6518. /// </summary>
  6519. /// <param name="index">Index to update.</param>
  6520. public void UpdateStickyPosition (int index) {
  6521. index = index%stickyPosition.Length;
  6522. if (collisionGameObject[index]!=null && collisionGameObject[index].activeSelf && collisionGameObject[index].activeInHierarchy) {
  6523. stickyPosition[index] = collisionTransform[index].TransformPoint(collisionTransformPosition[index]);
  6524. } else hasCollided[index] = false;
  6525. }
  6526. /// <summary>
  6527. /// Resets the specified index.
  6528. /// </summary>
  6529. /// <param name="index">Index.</param>
  6530. public void Reset (int index) {
  6531. index = index%hasCollided.Length;
  6532. hasCollided[index] = false;
  6533. }
  6534. /// <summary>
  6535. /// Clears out all the collisions. Use this if you want to toggle between having sticky and non-sticky behaviors.
  6536. /// </summary>
  6537. public void ClearCollisions () {
  6538. for(int i = 0; i<hasCollided.Length; i++)
  6539. hasCollided[i] = false;
  6540. }
  6541. /// <summary>
  6542. /// Copies this CollisionCache.
  6543. /// </summary>
  6544. public CollisionCache Clone () {
  6545. CollisionCache collisionCache = new CollisionCache(collisionPosition.Length);
  6546. collisionCache.hasCollided = (bool[])hasCollided.Clone();
  6547. collisionCache.collisionPosition = (Vector3[])collisionPosition.Clone();
  6548. collisionCache.collisionNormal = (Vector3[])collisionNormal.Clone();
  6549. collisionCache.collisionTransform = (Transform[])collisionTransform.Clone();
  6550. collisionCache.collisionTransformPosition = (Vector3[])collisionTransformPosition.Clone();
  6551. collisionCache.stickyPosition = (Vector3[])stickyPosition.Clone();
  6552. return collisionCache;
  6553. }
  6554. }
  6555. /// <summary>
  6556. /// The Simplex Noise algorithm applied in Turbulence.
  6557. /// </summary>
  6558. public class SimplexNoise {
  6559. // Simplex noise based on http://staffwww.itn.liu.se/~stegu/simplexnoise/simplexnoise.pdf for public domain by courtesy of Stefan Gustavson.
  6560. private static int[][] grad3 = new int[][] {
  6561. new int[] {1,1,0}, new int[] {-1,1,0}, new int[] {1,-1,0}, new int[] {-1,-1,0},
  6562. new int[] {1,0,1}, new int[] {-1,0,1}, new int[] {1,0,-1}, new int[] {-1,0,-1},
  6563. new int[] {0,1,1}, new int[] {0,-1,1}, new int[] {0,1,-1}, new int[] {0,-1,-1}};
  6564. private static int[][] grad4 = new int[][] {
  6565. new int[] {0,1,1,1}, new int[] {0,1,1,-1}, new int[] {0,1,-1,1}, new int[] {0,1,-1,-1},
  6566. new int[] {0,-1,1,1}, new int[] {0,-1,1,-1}, new int[] {0,-1,-1,1}, new int[] {0,-1,-1,-1},
  6567. new int[] {1,0,1,1}, new int[] {1,0,1,-1}, new int[] {1,0,-1,1}, new int[] {1,0,-1,-1},
  6568. new int[] {-1,0,1,1}, new int[] {-1,0,1,-1}, new int[] {-1,0,-1,1}, new int[] {-1,0,-1,-1},
  6569. new int[] {1,1,0,1}, new int[] {1,1,0,-1}, new int[] {1,-1,0,1}, new int[] {1,-1,0,-1},
  6570. new int[] {-1,1,0,1}, new int[] {-1,1,0,-1}, new int[] {-1,-1,0,1}, new int[] {-1,-1,0,-1},
  6571. new int[] {1,1,1,0}, new int[] {1,1,-1,0}, new int[] {1,-1,1,0}, new int[] {1,-1,-1,0},
  6572. new int[] {-1,1,1,0}, new int[] {-1,1,-1,0}, new int[] {-1,-1,1,0}, new int[] {-1,-1,-1,0}};
  6573. private static int[] p = {151,160,137,91,90,15,
  6574. 131,13,201,95,96,53,194,233,7,225,140,36,103,30,69,142,8,99,37,240,21,10,23,
  6575. 190, 6,148,247,120,234,75,0,26,197,62,94,252,219,203,117,35,11,32,57,177,33,
  6576. 88,237,149,56,87,174,20,125,136,171,168, 68,175,74,165,71,134,139,48,27,166,
  6577. 77,146,158,231,83,111,229,122,60,211,133,230,220,105,92,41,55,46,245,40,244,
  6578. 102,143,54, 65,25,63,161, 1,216,80,73,209,76,132,187,208, 89,18,169,200,196,
  6579. 135,130,116,188,159,86,164,100,109,198,173,186, 3,64,52,217,226,250,124,123,
  6580. 5,202,38,147,118,126,255,82,85,212,207,206,59,227,47,16,58,17,182,189,28,42,
  6581. 223,183,170,213,119,248,152, 2,44,154,163, 70,221,153,101,155,167, 43,172,9,
  6582. 129,22,39,253, 19,98,108,110,79,113,224,232,178,185, 112,104,218,246,97,228,
  6583. 251,34,242,193,238,210,144,12,191,179,162,241, 81,51,145,235,249,14,239,107,
  6584. 49,192,214, 31,181,199,106,157,184, 84,204,176,115,121,50,45,127, 4,150,254,
  6585. 138,236,205,93,222,114,67,29,24,72,243,141,128,195,78,66,215,61,156,180};
  6586. // To remove the need for index wrapping, double the permutation table length
  6587. private static int[] perm = new int[512];
  6588. static SimplexNoise() { for(int i=0; i<512; i++) perm[i]=p[i & 255]; } // moved to constructor
  6589. // A lookup table to traverse the simplex around a given point in 4D.
  6590. // Details can be found where this table is used, in the 4D noise method.
  6591. private static int[][] simplex = new int[][] {
  6592. new int[] {0,1,2,3}, new int[] {0,1,3,2}, new int[] {0,0,0,0}, new int[] {0,2,3,1}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {1,2,3,0},
  6593. new int[] {0,2,1,3}, new int[] {0,0,0,0}, new int[] {0,3,1,2}, new int[] {0,3,2,1}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {1,3,2,0},
  6594. new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0},
  6595. new int[] {1,2,0,3}, new int[] {0,0,0,0}, new int[] {1,3,0,2}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {2,3,0,1}, new int[] {2,3,1,0},
  6596. new int[] {1,0,2,3}, new int[] {1,0,3,2}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {2,0,3,1}, new int[] {0,0,0,0}, new int[] {2,1,3,0},
  6597. new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0},
  6598. new int[] {2,0,1,3}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {3,0,1,2}, new int[] {3,0,2,1}, new int[] {0,0,0,0}, new int[] {3,1,2,0},
  6599. new int[] {2,1,0,3}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {0,0,0,0}, new int[] {3,1,0,2}, new int[] {0,0,0,0}, new int[] {3,2,0,1}, new int[] {3,2,1,0}};
  6600. // This method is a *lot* faster than using (int)Mathf.floor(x)
  6601. private static int fastfloor(double x) {
  6602. return x>0 ? (int)x : (int)x-1;
  6603. }
  6604. private static double dot(int[] g, double x, double y) {
  6605. return g[0]*x + g[1]*y;
  6606. }
  6607. private static double dot(int[] g, double x, double y, double z) {
  6608. return g[0]*x + g[1]*y + g[2]*z;
  6609. }
  6610. private static double dot(int[] g, double x, double y, double z, double w) {
  6611. return g[0]*x + g[1]*y + g[2]*z + g[3]*w;
  6612. }
  6613. // 3D simplex noise
  6614. public double noise(double xin, double yin, double zin) {
  6615. double n0, n1, n2, n3; // Noise contributions from the four corners
  6616. // Skew the input space to determine which simplex cell we're in
  6617. double F3 = 1.0/3.0;
  6618. double s = (xin+yin+zin)*F3; // Very nice and simple skew factor for 3D
  6619. int i = fastfloor(xin+s);
  6620. int j = fastfloor(yin+s);
  6621. int k = fastfloor(zin+s);
  6622. double G3 = 1.0/6.0; // Very nice and simple unskew factor, too
  6623. double t = (i+j+k)*G3;
  6624. double X0 = i-t; // Unskew the cell origin back to (x,y,z) space
  6625. double Y0 = j-t;
  6626. double Z0 = k-t;
  6627. double x0 = xin-X0; // The x,y,z distances from the cell origin
  6628. double y0 = yin-Y0;
  6629. double z0 = zin-Z0;
  6630. // For the 3D case, the simplex shape is a slightly irregular tetrahedron.
  6631. // Determine which simplex we are in.
  6632. int i1, j1, k1; // Offsets for second corner of simplex in (i,j,k) coords
  6633. int i2, j2, k2; // Offsets for third corner of simplex in (i,j,k) coords
  6634. if(x0>=y0) {
  6635. if(y0>=z0)
  6636. { i1=1; j1=0; k1=0; i2=1; j2=1; k2=0; } // X Y Z order
  6637. else if(x0>=z0) { i1=1; j1=0; k1=0; i2=1; j2=0; k2=1; } // X Z Y order
  6638. else { i1=0; j1=0; k1=1; i2=1; j2=0; k2=1; } // Z X Y order
  6639. }
  6640. else { // x0<y0
  6641. if(y0<z0) { i1=0; j1=0; k1=1; i2=0; j2=1; k2=1; } // Z Y X order
  6642. else if(x0<z0) { i1=0; j1=1; k1=0; i2=0; j2=1; k2=1; } // Y Z X order
  6643. else { i1=0; j1=1; k1=0; i2=1; j2=1; k2=0; } // Y X Z order
  6644. }
  6645. // A step of (1,0,0) in (i,j,k) means a step of (1-c,-c,-c) in (x,y,z),
  6646. // a step of (0,1,0) in (i,j,k) means a step of (-c,1-c,-c) in (x,y,z), and
  6647. // a step of (0,0,1) in (i,j,k) means a step of (-c,-c,1-c) in (x,y,z), where
  6648. // c = 1/6.
  6649. double x1 = x0 - i1 + G3; // Offsets for second corner in (x,y,z) coords
  6650. double y1 = y0 - j1 + G3;
  6651. double z1 = z0 - k1 + G3;
  6652. double x2 = x0 - i2 + 2.0*G3; // Offsets for third corner in (x,y,z) coords
  6653. double y2 = y0 - j2 + 2.0*G3;
  6654. double z2 = z0 - k2 + 2.0*G3;
  6655. double x3 = x0 - 1.0 + 3.0*G3; // Offsets for last corner in (x,y,z) coords
  6656. double y3 = y0 - 1.0 + 3.0*G3;
  6657. double z3 = z0 - 1.0 + 3.0*G3;
  6658. // Work out the hashed gradient indices of the four simplex corners
  6659. int ii = i & 255;
  6660. int jj = j & 255;
  6661. int kk = k & 255;
  6662. int gi0 = perm[ii+perm[jj+perm[kk]]] % 12;
  6663. int gi1 = perm[ii+i1+perm[jj+j1+perm[kk+k1]]] % 12;
  6664. int gi2 = perm[ii+i2+perm[jj+j2+perm[kk+k2]]] % 12;
  6665. int gi3 = perm[ii+1+perm[jj+1+perm[kk+1]]] % 12;
  6666. // Calculate the contribution from the four corners
  6667. double t0 = 0.5 - x0*x0 - y0*y0 - z0*z0;
  6668. if(t0<0) n0 = 0.0;
  6669. else {
  6670. t0 *= t0;
  6671. n0 = t0 * t0 * dot(grad3[gi0], x0, y0, z0);
  6672. }
  6673. double t1 = 0.6 - x1*x1 - y1*y1 - z1*z1;
  6674. if(t1<0) n1 = 0.0;
  6675. else {
  6676. t1 *= t1;
  6677. n1 = t1 * t1 * dot(grad3[gi1], x1, y1, z1);
  6678. }
  6679. double t2 = 0.6 - x2*x2 - y2*y2 - z2*z2;
  6680. if(t2<0) n2 = 0.0;
  6681. else {
  6682. t2 *= t2;
  6683. n2 = t2 * t2 * dot(grad3[gi2], x2, y2, z2);
  6684. }
  6685. double t3 = 0.6 - x3*x3 - y3*y3 - z3*z3;
  6686. if(t3<0) n3 = 0.0;
  6687. else {
  6688. t3 *= t3;
  6689. n3 = t3 * t3 * dot(grad3[gi3], x3, y3, z3);
  6690. }
  6691. // Add contributions from each corner to get the final noise value.
  6692. // The result is scaled to stay just inside [-1,1]
  6693. return 32.0*(n0 + n1 + n2 + n3);
  6694. }
  6695. // 4D simplex noise
  6696. public double noise(double x, double y, double z, double w) {
  6697. // The skewing and unskewing factors are hairy again for the 4D case
  6698. double F4 = (Mathf.Sqrt(5.0f)-1.0)/4.0;
  6699. double G4 = (5.0-Mathf.Sqrt(5.0f))/20.0;
  6700. double n0, n1, n2, n3, n4; // Noise contributions from the five corners
  6701. // Skew the (x,y,z,w) space to determine which cell of 24 simplices we're in
  6702. double s = (x + y + z + w) * F4; // Factor for 4D skewing
  6703. int i = fastfloor(x + s);
  6704. int j = fastfloor(y + s);
  6705. int k = fastfloor(z + s);
  6706. int l = fastfloor(w + s);
  6707. double t = (i + j + k + l) * G4; // Factor for 4D unskewing
  6708. double X0 = i - t; // Unskew the cell origin back to (x,y,z,w) space
  6709. double Y0 = j - t;
  6710. double Z0 = k - t;
  6711. double W0 = l - t;
  6712. double x0 = x - X0; // The x,y,z,w distances from the cell origin
  6713. double y0 = y - Y0;
  6714. double z0 = z - Z0;
  6715. double w0 = w - W0;
  6716. // For the 4D case, the simplex is a 4D shape I won't even try to describe.
  6717. // To find out which of the 24 possible simplices we're in, we need to
  6718. // determine the magnitude ordering of x0, y0, z0 and w0.
  6719. // The method below is a good way of finding the ordering of x,y,z,w and
  6720. // then find the correct traversal order for the simplex we’re in.
  6721. // First, six pair-wise comparisons are performed between each possible pair
  6722. // of the four coordinates, and the results are used to add up binary bits
  6723. // for an integer index.
  6724. int c1 = (x0 > y0) ? 32 : 0;
  6725. int c2 = (x0 > z0) ? 16 : 0;
  6726. int c3 = (y0 > z0) ? 8 : 0;
  6727. int c4 = (x0 > w0) ? 4 : 0;
  6728. int c5 = (y0 > w0) ? 2 : 0;
  6729. int c6 = (z0 > w0) ? 1 : 0;
  6730. int c = c1 + c2 + c3 + c4 + c5 + c6;
  6731. int i1, j1, k1, l1; // The integer offsets for the second simplex corner
  6732. int i2, j2, k2, l2; // The integer offsets for the third simplex corner
  6733. int i3, j3, k3, l3; // The integer offsets for the fourth simplex corner
  6734. // simplex[c] is a 4-vector with the numbers 0, 1, 2 and 3 in some order.
  6735. // Many values of c will never occur, since e.g. x>y>z>w makes x<z, y<w and x<w
  6736. // impossible. Only the 24 indices which have non-zero entries make any sense.
  6737. // We use a thresholding to set the coordinates in turn from the largest magnitude.
  6738. // The number 3 in the "simplex" array is at the position of the largest coordinate.
  6739. i1 = simplex[c][0]>=3 ? 1 : 0;
  6740. j1 = simplex[c][1]>=3 ? 1 : 0;
  6741. k1 = simplex[c][2]>=3 ? 1 : 0;
  6742. l1 = simplex[c][3]>=3 ? 1 : 0;
  6743. // The number 2 in the "simplex" array is at the second largest coordinate.
  6744. i2 = simplex[c][0]>=2 ? 1 : 0;
  6745. j2 = simplex[c][1]>=2 ? 1 : 0; k2 = simplex[c][2]>=2 ? 1 : 0;
  6746. l2 = simplex[c][3]>=2 ? 1 : 0;
  6747. // The number 1 in the "simplex" array is at the second smallest coordinate.
  6748. i3 = simplex[c][0]>=1 ? 1 : 0;
  6749. j3 = simplex[c][1]>=1 ? 1 : 0;
  6750. k3 = simplex[c][2]>=1 ? 1 : 0;
  6751. l3 = simplex[c][3]>=1 ? 1 : 0;
  6752. // The fifth corner has all coordinate offsets = 1, so no need to look that up.
  6753. double x1 = x0 - i1 + G4; // Offsets for second corner in (x,y,z,w) coords
  6754. double y1 = y0 - j1 + G4;
  6755. double z1 = z0 - k1 + G4;
  6756. double w1 = w0 - l1 + G4;
  6757. double x2 = x0 - i2 + 2.0*G4; // Offsets for third corner in (x,y,z,w) coords
  6758. double y2 = y0 - j2 + 2.0*G4;
  6759. double z2 = z0 - k2 + 2.0*G4;
  6760. double w2 = w0 - l2 + 2.0*G4;
  6761. double x3 = x0 - i3 + 3.0*G4; // Offsets for fourth corner in (x,y,z,w) coords
  6762. double y3 = y0 - j3 + 3.0*G4;
  6763. double z3 = z0 - k3 + 3.0*G4;
  6764. double w3 = w0 - l3 + 3.0*G4;
  6765. double x4 = x0 - 1.0 + 4.0*G4; // Offsets for last corner in (x,y,z,w) coords
  6766. double y4 = y0 - 1.0 + 4.0*G4;
  6767. double z4 = z0 - 1.0 + 4.0*G4;
  6768. double w4 = w0 - 1.0 + 4.0*G4;
  6769. // Work out the hashed gradient indices of the five simplex corners
  6770. int ii = i & 255;
  6771. int jj = j & 255;
  6772. int kk = k & 255;
  6773. int ll = l & 255;
  6774. int gi0 = perm[ii+perm[jj+perm[kk+perm[ll]]]] % 32;
  6775. int gi1 = perm[ii+i1+perm[jj+j1+perm[kk+k1+perm[ll+l1]]]] % 32;
  6776. int gi2 = perm[ii+i2+perm[jj+j2+perm[kk+k2+perm[ll+l2]]]] % 32;
  6777. int gi3 = perm[ii+i3+perm[jj+j3+perm[kk+k3+perm[ll+l3]]]] % 32;
  6778. int gi4 = perm[ii+1+perm[jj+1+perm[kk+1+perm[ll+1]]]] % 32;
  6779. // Calculate the contribution from the five corners
  6780. double t0 = 0.5 - x0*x0 - y0*y0 - z0*z0 - w0*w0;
  6781. if(t0<0) n0 = 0.0;
  6782. else {
  6783. t0 *= t0;
  6784. n0 = t0 * t0 * dot(grad4[gi0], x0, y0, z0, w0);
  6785. }
  6786. double t1 = 0.6 - x1*x1 - y1*y1 - z1*z1 - w1*w1;
  6787. if(t1<0) n1 = 0.0;
  6788. else {
  6789. t1 *= t1;
  6790. n1 = t1 * t1 * dot(grad4[gi1], x1, y1, z1, w1);
  6791. }
  6792. double t2 = 0.6 - x2*x2 - y2*y2 - z2*z2 - w2*w2;
  6793. if(t2<0) n2 = 0.0;
  6794. else {
  6795. t2 *= t2;
  6796. n2 = t2 * t2 * dot(grad4[gi2], x2, y2, z2, w2);
  6797. } double t3 = 0.6 - x3*x3 - y3*y3 - z3*z3 - w3*w3;
  6798. if(t3<0) n3 = 0.0;
  6799. else {
  6800. t3 *= t3;
  6801. n3 = t3 * t3 * dot(grad4[gi3], x3, y3, z3, w3);
  6802. }
  6803. double t4 = 0.6 - x4*x4 - y4*y4 - z4*z4 - w4*w4;
  6804. if(t4<0) n4 = 0.0;
  6805. else {
  6806. t4 *= t4;
  6807. n4 = t4 * t4 * dot(grad4[gi4], x4, y4, z4, w4);
  6808. }
  6809. // Sum up and scale the result to cover the range [-1,1]
  6810. return 27.0 * (n0 + n1 + n2 + n3 + n4);
  6811. }
  6812. }
  6813. }