GeometryPipeline-466ad516.js 115 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708
  1. /**
  2. * Cesium - https://github.com/CesiumGS/cesium
  3. *
  4. * Copyright 2011-2020 Cesium Contributors
  5. *
  6. * Licensed under the Apache License, Version 2.0 (the "License");
  7. * you may not use this file except in compliance with the License.
  8. * You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. *
  18. * Columbus View (Pat. Pend.)
  19. *
  20. * Portions licensed separately.
  21. * See https://github.com/CesiumGS/cesium/blob/master/LICENSE.md for full licensing details.
  22. */
  23. define(['exports', './when-54c2dc71', './Check-6c0211bc', './Math-1124a290', './Cartesian2-33d2657c', './Transforms-8be64844', './ComponentDatatype-a26dd044', './GeometryAttribute-e9a8b203', './AttributeCompression-75249b5e', './EncodedCartesian3-6c97231d', './IndexDatatype-25023891', './IntersectionTests-afc38163', './Plane-fa30fc46'], function (exports, when, Check, _Math, Cartesian2, Transforms, ComponentDatatype, GeometryAttribute, AttributeCompression, EncodedCartesian3, IndexDatatype, IntersectionTests, Plane) { 'use strict';
  24. var scratchCartesian1 = new Cartesian2.Cartesian3();
  25. var scratchCartesian2 = new Cartesian2.Cartesian3();
  26. var scratchCartesian3 = new Cartesian2.Cartesian3();
  27. /**
  28. * Computes the barycentric coordinates for a point with respect to a triangle.
  29. *
  30. * @function
  31. *
  32. * @param {Cartesian2|Cartesian3} point The point to test.
  33. * @param {Cartesian2|Cartesian3} p0 The first point of the triangle, corresponding to the barycentric x-axis.
  34. * @param {Cartesian2|Cartesian3} p1 The second point of the triangle, corresponding to the barycentric y-axis.
  35. * @param {Cartesian2|Cartesian3} p2 The third point of the triangle, corresponding to the barycentric z-axis.
  36. * @param {Cartesian3} [result] The object onto which to store the result.
  37. * @returns {Cartesian3} The modified result parameter or a new Cartesian3 instance if one was not provided.
  38. *
  39. * @example
  40. * // Returns Cartesian3.UNIT_X
  41. * var p = new Cesium.Cartesian3(-1.0, 0.0, 0.0);
  42. * var b = Cesium.barycentricCoordinates(p,
  43. * new Cesium.Cartesian3(-1.0, 0.0, 0.0),
  44. * new Cesium.Cartesian3( 1.0, 0.0, 0.0),
  45. * new Cesium.Cartesian3( 0.0, 1.0, 1.0));
  46. */
  47. function barycentricCoordinates(point, p0, p1, p2, result) {
  48. //>>includeStart('debug', pragmas.debug);
  49. Check.Check.defined("point", point);
  50. Check.Check.defined("p0", p0);
  51. Check.Check.defined("p1", p1);
  52. Check.Check.defined("p2", p2);
  53. //>>includeEnd('debug');
  54. if (!when.defined(result)) {
  55. result = new Cartesian2.Cartesian3();
  56. }
  57. // Implementation based on http://www.blackpawn.com/texts/pointinpoly/default.html.
  58. var v0;
  59. var v1;
  60. var v2;
  61. var dot00;
  62. var dot01;
  63. var dot02;
  64. var dot11;
  65. var dot12;
  66. if (!when.defined(p0.z)) {
  67. if (Cartesian2.Cartesian2.equalsEpsilon(point, p0, _Math.CesiumMath.EPSILON14)) {
  68. return Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.UNIT_X, result);
  69. }
  70. if (Cartesian2.Cartesian2.equalsEpsilon(point, p1, _Math.CesiumMath.EPSILON14)) {
  71. return Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.UNIT_Y, result);
  72. }
  73. if (Cartesian2.Cartesian2.equalsEpsilon(point, p2, _Math.CesiumMath.EPSILON14)) {
  74. return Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.UNIT_Z, result);
  75. }
  76. v0 = Cartesian2.Cartesian2.subtract(p1, p0, scratchCartesian1);
  77. v1 = Cartesian2.Cartesian2.subtract(p2, p0, scratchCartesian2);
  78. v2 = Cartesian2.Cartesian2.subtract(point, p0, scratchCartesian3);
  79. dot00 = Cartesian2.Cartesian2.dot(v0, v0);
  80. dot01 = Cartesian2.Cartesian2.dot(v0, v1);
  81. dot02 = Cartesian2.Cartesian2.dot(v0, v2);
  82. dot11 = Cartesian2.Cartesian2.dot(v1, v1);
  83. dot12 = Cartesian2.Cartesian2.dot(v1, v2);
  84. } else {
  85. if (Cartesian2.Cartesian3.equalsEpsilon(point, p0, _Math.CesiumMath.EPSILON14)) {
  86. return Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.UNIT_X, result);
  87. }
  88. if (Cartesian2.Cartesian3.equalsEpsilon(point, p1, _Math.CesiumMath.EPSILON14)) {
  89. return Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.UNIT_Y, result);
  90. }
  91. if (Cartesian2.Cartesian3.equalsEpsilon(point, p2, _Math.CesiumMath.EPSILON14)) {
  92. return Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.UNIT_Z, result);
  93. }
  94. v0 = Cartesian2.Cartesian3.subtract(p1, p0, scratchCartesian1);
  95. v1 = Cartesian2.Cartesian3.subtract(p2, p0, scratchCartesian2);
  96. v2 = Cartesian2.Cartesian3.subtract(point, p0, scratchCartesian3);
  97. dot00 = Cartesian2.Cartesian3.dot(v0, v0);
  98. dot01 = Cartesian2.Cartesian3.dot(v0, v1);
  99. dot02 = Cartesian2.Cartesian3.dot(v0, v2);
  100. dot11 = Cartesian2.Cartesian3.dot(v1, v1);
  101. dot12 = Cartesian2.Cartesian3.dot(v1, v2);
  102. }
  103. result.y = dot11 * dot02 - dot01 * dot12;
  104. result.z = dot00 * dot12 - dot01 * dot02;
  105. var q = dot00 * dot11 - dot01 * dot01;
  106. // This is done to avoid dividing by infinity causing a NaN
  107. if (result.y !== 0) {
  108. result.y /= q;
  109. }
  110. if (result.z !== 0) {
  111. result.z /= q;
  112. }
  113. result.x = 1.0 - result.y - result.z;
  114. return result;
  115. }
  116. /**
  117. * Encapsulates an algorithm to optimize triangles for the post
  118. * vertex-shader cache. This is based on the 2007 SIGGRAPH paper
  119. * 'Fast Triangle Reordering for Vertex Locality and Reduced Overdraw.'
  120. * The runtime is linear but several passes are made.
  121. *
  122. * @namespace Tipsify
  123. *
  124. * @see <a href='http://gfx.cs.princeton.edu/pubs/Sander_2007_%3ETR/tipsy.pdf'>
  125. * Fast Triangle Reordering for Vertex Locality and Reduced Overdraw</a>
  126. * by Sander, Nehab, and Barczak
  127. *
  128. * @private
  129. */
  130. var Tipsify = {};
  131. /**
  132. * Calculates the average cache miss ratio (ACMR) for a given set of indices.
  133. *
  134. * @param {Object} options Object with the following properties:
  135. * @param {Number[]} options.indices Lists triads of numbers corresponding to the indices of the vertices
  136. * in the vertex buffer that define the geometry's triangles.
  137. * @param {Number} [options.maximumIndex] The maximum value of the elements in <code>args.indices</code>.
  138. * If not supplied, this value will be computed.
  139. * @param {Number} [options.cacheSize=24] The number of vertices that can be stored in the cache at any one time.
  140. * @returns {Number} The average cache miss ratio (ACMR).
  141. *
  142. * @exception {DeveloperError} indices length must be a multiple of three.
  143. * @exception {DeveloperError} cacheSize must be greater than two.
  144. *
  145. * @example
  146. * var indices = [0, 1, 2, 3, 4, 5];
  147. * var maxIndex = 5;
  148. * var cacheSize = 3;
  149. * var acmr = Cesium.Tipsify.calculateACMR({indices : indices, maxIndex : maxIndex, cacheSize : cacheSize});
  150. */
  151. Tipsify.calculateACMR = function (options) {
  152. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  153. var indices = options.indices;
  154. var maximumIndex = options.maximumIndex;
  155. var cacheSize = when.defaultValue(options.cacheSize, 24);
  156. //>>includeStart('debug', pragmas.debug);
  157. if (!when.defined(indices)) {
  158. throw new Check.DeveloperError("indices is required.");
  159. }
  160. //>>includeEnd('debug');
  161. var numIndices = indices.length;
  162. //>>includeStart('debug', pragmas.debug);
  163. if (numIndices < 3 || numIndices % 3 !== 0) {
  164. throw new Check.DeveloperError("indices length must be a multiple of three.");
  165. }
  166. if (maximumIndex <= 0) {
  167. throw new Check.DeveloperError("maximumIndex must be greater than zero.");
  168. }
  169. if (cacheSize < 3) {
  170. throw new Check.DeveloperError("cacheSize must be greater than two.");
  171. }
  172. //>>includeEnd('debug');
  173. // Compute the maximumIndex if not given
  174. if (!when.defined(maximumIndex)) {
  175. maximumIndex = 0;
  176. var currentIndex = 0;
  177. var intoIndices = indices[currentIndex];
  178. while (currentIndex < numIndices) {
  179. if (intoIndices > maximumIndex) {
  180. maximumIndex = intoIndices;
  181. }
  182. ++currentIndex;
  183. intoIndices = indices[currentIndex];
  184. }
  185. }
  186. // Vertex time stamps
  187. var vertexTimeStamps = [];
  188. for (var i = 0; i < maximumIndex + 1; i++) {
  189. vertexTimeStamps[i] = 0;
  190. }
  191. // Cache processing
  192. var s = cacheSize + 1;
  193. for (var j = 0; j < numIndices; ++j) {
  194. if (s - vertexTimeStamps[indices[j]] > cacheSize) {
  195. vertexTimeStamps[indices[j]] = s;
  196. ++s;
  197. }
  198. }
  199. return (s - cacheSize + 1) / (numIndices / 3);
  200. };
  201. /**
  202. * Optimizes triangles for the post-vertex shader cache.
  203. *
  204. * @param {Object} options Object with the following properties:
  205. * @param {Number[]} options.indices Lists triads of numbers corresponding to the indices of the vertices
  206. * in the vertex buffer that define the geometry's triangles.
  207. * @param {Number} [options.maximumIndex] The maximum value of the elements in <code>args.indices</code>.
  208. * If not supplied, this value will be computed.
  209. * @param {Number} [options.cacheSize=24] The number of vertices that can be stored in the cache at any one time.
  210. * @returns {Number[]} A list of the input indices in an optimized order.
  211. *
  212. * @exception {DeveloperError} indices length must be a multiple of three.
  213. * @exception {DeveloperError} cacheSize must be greater than two.
  214. *
  215. * @example
  216. * var indices = [0, 1, 2, 3, 4, 5];
  217. * var maxIndex = 5;
  218. * var cacheSize = 3;
  219. * var reorderedIndices = Cesium.Tipsify.tipsify({indices : indices, maxIndex : maxIndex, cacheSize : cacheSize});
  220. */
  221. Tipsify.tipsify = function (options) {
  222. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  223. var indices = options.indices;
  224. var maximumIndex = options.maximumIndex;
  225. var cacheSize = when.defaultValue(options.cacheSize, 24);
  226. var cursor;
  227. function skipDeadEnd(vertices, deadEnd, indices, maximumIndexPlusOne) {
  228. while (deadEnd.length >= 1) {
  229. // while the stack is not empty
  230. var d = deadEnd[deadEnd.length - 1]; // top of the stack
  231. deadEnd.splice(deadEnd.length - 1, 1); // pop the stack
  232. if (vertices[d].numLiveTriangles > 0) {
  233. return d;
  234. }
  235. }
  236. while (cursor < maximumIndexPlusOne) {
  237. if (vertices[cursor].numLiveTriangles > 0) {
  238. ++cursor;
  239. return cursor - 1;
  240. }
  241. ++cursor;
  242. }
  243. return -1;
  244. }
  245. function getNextVertex(
  246. indices,
  247. cacheSize,
  248. oneRing,
  249. vertices,
  250. s,
  251. deadEnd,
  252. maximumIndexPlusOne
  253. ) {
  254. var n = -1;
  255. var p;
  256. var m = -1;
  257. var itOneRing = 0;
  258. while (itOneRing < oneRing.length) {
  259. var index = oneRing[itOneRing];
  260. if (vertices[index].numLiveTriangles) {
  261. p = 0;
  262. if (
  263. s -
  264. vertices[index].timeStamp +
  265. 2 * vertices[index].numLiveTriangles <=
  266. cacheSize
  267. ) {
  268. p = s - vertices[index].timeStamp;
  269. }
  270. if (p > m || m === -1) {
  271. m = p;
  272. n = index;
  273. }
  274. }
  275. ++itOneRing;
  276. }
  277. if (n === -1) {
  278. return skipDeadEnd(vertices, deadEnd, indices, maximumIndexPlusOne);
  279. }
  280. return n;
  281. }
  282. //>>includeStart('debug', pragmas.debug);
  283. if (!when.defined(indices)) {
  284. throw new Check.DeveloperError("indices is required.");
  285. }
  286. //>>includeEnd('debug');
  287. var numIndices = indices.length;
  288. //>>includeStart('debug', pragmas.debug);
  289. if (numIndices < 3 || numIndices % 3 !== 0) {
  290. throw new Check.DeveloperError("indices length must be a multiple of three.");
  291. }
  292. if (maximumIndex <= 0) {
  293. throw new Check.DeveloperError("maximumIndex must be greater than zero.");
  294. }
  295. if (cacheSize < 3) {
  296. throw new Check.DeveloperError("cacheSize must be greater than two.");
  297. }
  298. //>>includeEnd('debug');
  299. // Determine maximum index
  300. var maximumIndexPlusOne = 0;
  301. var currentIndex = 0;
  302. var intoIndices = indices[currentIndex];
  303. var endIndex = numIndices;
  304. if (when.defined(maximumIndex)) {
  305. maximumIndexPlusOne = maximumIndex + 1;
  306. } else {
  307. while (currentIndex < endIndex) {
  308. if (intoIndices > maximumIndexPlusOne) {
  309. maximumIndexPlusOne = intoIndices;
  310. }
  311. ++currentIndex;
  312. intoIndices = indices[currentIndex];
  313. }
  314. if (maximumIndexPlusOne === -1) {
  315. return 0;
  316. }
  317. ++maximumIndexPlusOne;
  318. }
  319. // Vertices
  320. var vertices = [];
  321. var i;
  322. for (i = 0; i < maximumIndexPlusOne; i++) {
  323. vertices[i] = {
  324. numLiveTriangles: 0,
  325. timeStamp: 0,
  326. vertexTriangles: [],
  327. };
  328. }
  329. currentIndex = 0;
  330. var triangle = 0;
  331. while (currentIndex < endIndex) {
  332. vertices[indices[currentIndex]].vertexTriangles.push(triangle);
  333. ++vertices[indices[currentIndex]].numLiveTriangles;
  334. vertices[indices[currentIndex + 1]].vertexTriangles.push(triangle);
  335. ++vertices[indices[currentIndex + 1]].numLiveTriangles;
  336. vertices[indices[currentIndex + 2]].vertexTriangles.push(triangle);
  337. ++vertices[indices[currentIndex + 2]].numLiveTriangles;
  338. ++triangle;
  339. currentIndex += 3;
  340. }
  341. // Starting index
  342. var f = 0;
  343. // Time Stamp
  344. var s = cacheSize + 1;
  345. cursor = 1;
  346. // Process
  347. var oneRing = [];
  348. var deadEnd = []; //Stack
  349. var vertex;
  350. var intoVertices;
  351. var currentOutputIndex = 0;
  352. var outputIndices = [];
  353. var numTriangles = numIndices / 3;
  354. var triangleEmitted = [];
  355. for (i = 0; i < numTriangles; i++) {
  356. triangleEmitted[i] = false;
  357. }
  358. var index;
  359. var limit;
  360. while (f !== -1) {
  361. oneRing = [];
  362. intoVertices = vertices[f];
  363. limit = intoVertices.vertexTriangles.length;
  364. for (var k = 0; k < limit; ++k) {
  365. triangle = intoVertices.vertexTriangles[k];
  366. if (!triangleEmitted[triangle]) {
  367. triangleEmitted[triangle] = true;
  368. currentIndex = triangle + triangle + triangle;
  369. for (var j = 0; j < 3; ++j) {
  370. // Set this index as a possible next index
  371. index = indices[currentIndex];
  372. oneRing.push(index);
  373. deadEnd.push(index);
  374. // Output index
  375. outputIndices[currentOutputIndex] = index;
  376. ++currentOutputIndex;
  377. // Cache processing
  378. vertex = vertices[index];
  379. --vertex.numLiveTriangles;
  380. if (s - vertex.timeStamp > cacheSize) {
  381. vertex.timeStamp = s;
  382. ++s;
  383. }
  384. ++currentIndex;
  385. }
  386. }
  387. }
  388. f = getNextVertex(
  389. indices,
  390. cacheSize,
  391. oneRing,
  392. vertices,
  393. s,
  394. deadEnd,
  395. maximumIndexPlusOne
  396. );
  397. }
  398. return outputIndices;
  399. };
  400. /**
  401. * Content pipeline functions for geometries.
  402. *
  403. * @namespace GeometryPipeline
  404. *
  405. * @see Geometry
  406. */
  407. var GeometryPipeline = {};
  408. function addTriangle(lines, index, i0, i1, i2) {
  409. lines[index++] = i0;
  410. lines[index++] = i1;
  411. lines[index++] = i1;
  412. lines[index++] = i2;
  413. lines[index++] = i2;
  414. lines[index] = i0;
  415. }
  416. function trianglesToLines(triangles) {
  417. var count = triangles.length;
  418. var size = (count / 3) * 6;
  419. var lines = IndexDatatype.IndexDatatype.createTypedArray(count, size);
  420. var index = 0;
  421. for (var i = 0; i < count; i += 3, index += 6) {
  422. addTriangle(lines, index, triangles[i], triangles[i + 1], triangles[i + 2]);
  423. }
  424. return lines;
  425. }
  426. function triangleStripToLines(triangles) {
  427. var count = triangles.length;
  428. if (count >= 3) {
  429. var size = (count - 2) * 6;
  430. var lines = IndexDatatype.IndexDatatype.createTypedArray(count, size);
  431. addTriangle(lines, 0, triangles[0], triangles[1], triangles[2]);
  432. var index = 6;
  433. for (var i = 3; i < count; ++i, index += 6) {
  434. addTriangle(
  435. lines,
  436. index,
  437. triangles[i - 1],
  438. triangles[i],
  439. triangles[i - 2]
  440. );
  441. }
  442. return lines;
  443. }
  444. return new Uint16Array();
  445. }
  446. function triangleFanToLines(triangles) {
  447. if (triangles.length > 0) {
  448. var count = triangles.length - 1;
  449. var size = (count - 1) * 6;
  450. var lines = IndexDatatype.IndexDatatype.createTypedArray(count, size);
  451. var base = triangles[0];
  452. var index = 0;
  453. for (var i = 1; i < count; ++i, index += 6) {
  454. addTriangle(lines, index, base, triangles[i], triangles[i + 1]);
  455. }
  456. return lines;
  457. }
  458. return new Uint16Array();
  459. }
  460. /**
  461. * Converts a geometry's triangle indices to line indices. If the geometry has an <code>indices</code>
  462. * and its <code>primitiveType</code> is <code>TRIANGLES</code>, <code>TRIANGLE_STRIP</code>,
  463. * <code>TRIANGLE_FAN</code>, it is converted to <code>LINES</code>; otherwise, the geometry is not changed.
  464. * <p>
  465. * This is commonly used to create a wireframe geometry for visual debugging.
  466. * </p>
  467. *
  468. * @param {Geometry} geometry The geometry to modify.
  469. * @returns {Geometry} The modified <code>geometry</code> argument, with its triangle indices converted to lines.
  470. *
  471. * @exception {DeveloperError} geometry.primitiveType must be TRIANGLES, TRIANGLE_STRIP, or TRIANGLE_FAN.
  472. *
  473. * @example
  474. * geometry = Cesium.GeometryPipeline.toWireframe(geometry);
  475. */
  476. GeometryPipeline.toWireframe = function (geometry) {
  477. //>>includeStart('debug', pragmas.debug);
  478. if (!when.defined(geometry)) {
  479. throw new Check.DeveloperError("geometry is required.");
  480. }
  481. //>>includeEnd('debug');
  482. var indices = geometry.indices;
  483. if (when.defined(indices)) {
  484. switch (geometry.primitiveType) {
  485. case GeometryAttribute.PrimitiveType.TRIANGLES:
  486. geometry.indices = trianglesToLines(indices);
  487. break;
  488. case GeometryAttribute.PrimitiveType.TRIANGLE_STRIP:
  489. geometry.indices = triangleStripToLines(indices);
  490. break;
  491. case GeometryAttribute.PrimitiveType.TRIANGLE_FAN:
  492. geometry.indices = triangleFanToLines(indices);
  493. break;
  494. //>>includeStart('debug', pragmas.debug);
  495. default:
  496. throw new Check.DeveloperError(
  497. "geometry.primitiveType must be TRIANGLES, TRIANGLE_STRIP, or TRIANGLE_FAN."
  498. );
  499. //>>includeEnd('debug');
  500. }
  501. geometry.primitiveType = GeometryAttribute.PrimitiveType.LINES;
  502. }
  503. return geometry;
  504. };
  505. /**
  506. * Creates a new {@link Geometry} with <code>LINES</code> representing the provided
  507. * attribute (<code>attributeName</code>) for the provided geometry. This is used to
  508. * visualize vector attributes like normals, tangents, and bitangents.
  509. *
  510. * @param {Geometry} geometry The <code>Geometry</code> instance with the attribute.
  511. * @param {String} [attributeName='normal'] The name of the attribute.
  512. * @param {Number} [length=10000.0] The length of each line segment in meters. This can be negative to point the vector in the opposite direction.
  513. * @returns {Geometry} A new <code>Geometry</code> instance with line segments for the vector.
  514. *
  515. * @exception {DeveloperError} geometry.attributes must have an attribute with the same name as the attributeName parameter.
  516. *
  517. * @example
  518. * var geometry = Cesium.GeometryPipeline.createLineSegmentsForVectors(instance.geometry, 'bitangent', 100000.0);
  519. */
  520. GeometryPipeline.createLineSegmentsForVectors = function (
  521. geometry,
  522. attributeName,
  523. length
  524. ) {
  525. attributeName = when.defaultValue(attributeName, "normal");
  526. //>>includeStart('debug', pragmas.debug);
  527. if (!when.defined(geometry)) {
  528. throw new Check.DeveloperError("geometry is required.");
  529. }
  530. if (!when.defined(geometry.attributes.position)) {
  531. throw new Check.DeveloperError("geometry.attributes.position is required.");
  532. }
  533. if (!when.defined(geometry.attributes[attributeName])) {
  534. throw new Check.DeveloperError(
  535. "geometry.attributes must have an attribute with the same name as the attributeName parameter, " +
  536. attributeName +
  537. "."
  538. );
  539. }
  540. //>>includeEnd('debug');
  541. length = when.defaultValue(length, 10000.0);
  542. var positions = geometry.attributes.position.values;
  543. var vectors = geometry.attributes[attributeName].values;
  544. var positionsLength = positions.length;
  545. var newPositions = new Float64Array(2 * positionsLength);
  546. var j = 0;
  547. for (var i = 0; i < positionsLength; i += 3) {
  548. newPositions[j++] = positions[i];
  549. newPositions[j++] = positions[i + 1];
  550. newPositions[j++] = positions[i + 2];
  551. newPositions[j++] = positions[i] + vectors[i] * length;
  552. newPositions[j++] = positions[i + 1] + vectors[i + 1] * length;
  553. newPositions[j++] = positions[i + 2] + vectors[i + 2] * length;
  554. }
  555. var newBoundingSphere;
  556. var bs = geometry.boundingSphere;
  557. if (when.defined(bs)) {
  558. newBoundingSphere = new Transforms.BoundingSphere(bs.center, bs.radius + length);
  559. }
  560. return new GeometryAttribute.Geometry({
  561. attributes: {
  562. position: new GeometryAttribute.GeometryAttribute({
  563. componentDatatype: ComponentDatatype.ComponentDatatype.DOUBLE,
  564. componentsPerAttribute: 3,
  565. values: newPositions,
  566. }),
  567. },
  568. primitiveType: GeometryAttribute.PrimitiveType.LINES,
  569. boundingSphere: newBoundingSphere,
  570. });
  571. };
  572. /**
  573. * Creates an object that maps attribute names to unique locations (indices)
  574. * for matching vertex attributes and shader programs.
  575. *
  576. * @param {Geometry} geometry The geometry, which is not modified, to create the object for.
  577. * @returns {Object} An object with attribute name / index pairs.
  578. *
  579. * @example
  580. * var attributeLocations = Cesium.GeometryPipeline.createAttributeLocations(geometry);
  581. * // Example output
  582. * // {
  583. * // 'position' : 0,
  584. * // 'normal' : 1
  585. * // }
  586. */
  587. GeometryPipeline.createAttributeLocations = function (geometry) {
  588. //>>includeStart('debug', pragmas.debug);
  589. if (!when.defined(geometry)) {
  590. throw new Check.DeveloperError("geometry is required.");
  591. }
  592. //>>includeEnd('debug')
  593. // There can be a WebGL performance hit when attribute 0 is disabled, so
  594. // assign attribute locations to well-known attributes.
  595. var semantics = [
  596. "position",
  597. "positionHigh",
  598. "positionLow",
  599. // From VertexFormat.position - after 2D projection and high-precision encoding
  600. "position3DHigh",
  601. "position3DLow",
  602. "position2DHigh",
  603. "position2DLow",
  604. // From Primitive
  605. "pickColor",
  606. // From VertexFormat
  607. "normal",
  608. "st",
  609. "tangent",
  610. "bitangent",
  611. // For shadow volumes
  612. "extrudeDirection",
  613. // From compressing texture coordinates and normals
  614. "compressedAttributes",
  615. ];
  616. var attributes = geometry.attributes;
  617. var indices = {};
  618. var j = 0;
  619. var i;
  620. var len = semantics.length;
  621. // Attribute locations for well-known attributes
  622. for (i = 0; i < len; ++i) {
  623. var semantic = semantics[i];
  624. if (when.defined(attributes[semantic])) {
  625. indices[semantic] = j++;
  626. }
  627. }
  628. // Locations for custom attributes
  629. for (var name in attributes) {
  630. if (attributes.hasOwnProperty(name) && !when.defined(indices[name])) {
  631. indices[name] = j++;
  632. }
  633. }
  634. return indices;
  635. };
  636. /**
  637. * Reorders a geometry's attributes and <code>indices</code> to achieve better performance from the GPU's pre-vertex-shader cache.
  638. *
  639. * @param {Geometry} geometry The geometry to modify.
  640. * @returns {Geometry} The modified <code>geometry</code> argument, with its attributes and indices reordered for the GPU's pre-vertex-shader cache.
  641. *
  642. * @exception {DeveloperError} Each attribute array in geometry.attributes must have the same number of attributes.
  643. *
  644. *
  645. * @example
  646. * geometry = Cesium.GeometryPipeline.reorderForPreVertexCache(geometry);
  647. *
  648. * @see GeometryPipeline.reorderForPostVertexCache
  649. */
  650. GeometryPipeline.reorderForPreVertexCache = function (geometry) {
  651. //>>includeStart('debug', pragmas.debug);
  652. if (!when.defined(geometry)) {
  653. throw new Check.DeveloperError("geometry is required.");
  654. }
  655. //>>includeEnd('debug');
  656. var numVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  657. var indices = geometry.indices;
  658. if (when.defined(indices)) {
  659. var indexCrossReferenceOldToNew = new Int32Array(numVertices);
  660. for (var i = 0; i < numVertices; i++) {
  661. indexCrossReferenceOldToNew[i] = -1;
  662. }
  663. // Construct cross reference and reorder indices
  664. var indicesIn = indices;
  665. var numIndices = indicesIn.length;
  666. var indicesOut = IndexDatatype.IndexDatatype.createTypedArray(numVertices, numIndices);
  667. var intoIndicesIn = 0;
  668. var intoIndicesOut = 0;
  669. var nextIndex = 0;
  670. var tempIndex;
  671. while (intoIndicesIn < numIndices) {
  672. tempIndex = indexCrossReferenceOldToNew[indicesIn[intoIndicesIn]];
  673. if (tempIndex !== -1) {
  674. indicesOut[intoIndicesOut] = tempIndex;
  675. } else {
  676. tempIndex = indicesIn[intoIndicesIn];
  677. indexCrossReferenceOldToNew[tempIndex] = nextIndex;
  678. indicesOut[intoIndicesOut] = nextIndex;
  679. ++nextIndex;
  680. }
  681. ++intoIndicesIn;
  682. ++intoIndicesOut;
  683. }
  684. geometry.indices = indicesOut;
  685. // Reorder attributes
  686. var attributes = geometry.attributes;
  687. for (var property in attributes) {
  688. if (
  689. attributes.hasOwnProperty(property) &&
  690. when.defined(attributes[property]) &&
  691. when.defined(attributes[property].values)
  692. ) {
  693. var attribute = attributes[property];
  694. var elementsIn = attribute.values;
  695. var intoElementsIn = 0;
  696. var numComponents = attribute.componentsPerAttribute;
  697. var elementsOut = ComponentDatatype.ComponentDatatype.createTypedArray(
  698. attribute.componentDatatype,
  699. nextIndex * numComponents
  700. );
  701. while (intoElementsIn < numVertices) {
  702. var temp = indexCrossReferenceOldToNew[intoElementsIn];
  703. if (temp !== -1) {
  704. for (var j = 0; j < numComponents; j++) {
  705. elementsOut[numComponents * temp + j] =
  706. elementsIn[numComponents * intoElementsIn + j];
  707. }
  708. }
  709. ++intoElementsIn;
  710. }
  711. attribute.values = elementsOut;
  712. }
  713. }
  714. }
  715. return geometry;
  716. };
  717. /**
  718. * Reorders a geometry's <code>indices</code> to achieve better performance from the GPU's
  719. * post vertex-shader cache by using the Tipsify algorithm. If the geometry <code>primitiveType</code>
  720. * is not <code>TRIANGLES</code> or the geometry does not have an <code>indices</code>, this function has no effect.
  721. *
  722. * @param {Geometry} geometry The geometry to modify.
  723. * @param {Number} [cacheCapacity=24] The number of vertices that can be held in the GPU's vertex cache.
  724. * @returns {Geometry} The modified <code>geometry</code> argument, with its indices reordered for the post-vertex-shader cache.
  725. *
  726. * @exception {DeveloperError} cacheCapacity must be greater than two.
  727. *
  728. *
  729. * @example
  730. * geometry = Cesium.GeometryPipeline.reorderForPostVertexCache(geometry);
  731. *
  732. * @see GeometryPipeline.reorderForPreVertexCache
  733. * @see {@link http://gfx.cs.princ0eton.edu/pubs/Sander_2007_%3ETR/tipsy.pdf|Fast Triangle Reordering for Vertex Locality and Reduced Overdraw}
  734. * by Sander, Nehab, and Barczak
  735. */
  736. GeometryPipeline.reorderForPostVertexCache = function (
  737. geometry,
  738. cacheCapacity
  739. ) {
  740. //>>includeStart('debug', pragmas.debug);
  741. if (!when.defined(geometry)) {
  742. throw new Check.DeveloperError("geometry is required.");
  743. }
  744. //>>includeEnd('debug');
  745. var indices = geometry.indices;
  746. if (geometry.primitiveType === GeometryAttribute.PrimitiveType.TRIANGLES && when.defined(indices)) {
  747. var numIndices = indices.length;
  748. var maximumIndex = 0;
  749. for (var j = 0; j < numIndices; j++) {
  750. if (indices[j] > maximumIndex) {
  751. maximumIndex = indices[j];
  752. }
  753. }
  754. geometry.indices = Tipsify.tipsify({
  755. indices: indices,
  756. maximumIndex: maximumIndex,
  757. cacheSize: cacheCapacity,
  758. });
  759. }
  760. return geometry;
  761. };
  762. function copyAttributesDescriptions(attributes) {
  763. var newAttributes = {};
  764. for (var attribute in attributes) {
  765. if (
  766. attributes.hasOwnProperty(attribute) &&
  767. when.defined(attributes[attribute]) &&
  768. when.defined(attributes[attribute].values)
  769. ) {
  770. var attr = attributes[attribute];
  771. newAttributes[attribute] = new GeometryAttribute.GeometryAttribute({
  772. componentDatatype: attr.componentDatatype,
  773. componentsPerAttribute: attr.componentsPerAttribute,
  774. normalize: attr.normalize,
  775. values: [],
  776. });
  777. }
  778. }
  779. return newAttributes;
  780. }
  781. function copyVertex(destinationAttributes, sourceAttributes, index) {
  782. for (var attribute in sourceAttributes) {
  783. if (
  784. sourceAttributes.hasOwnProperty(attribute) &&
  785. when.defined(sourceAttributes[attribute]) &&
  786. when.defined(sourceAttributes[attribute].values)
  787. ) {
  788. var attr = sourceAttributes[attribute];
  789. for (var k = 0; k < attr.componentsPerAttribute; ++k) {
  790. destinationAttributes[attribute].values.push(
  791. attr.values[index * attr.componentsPerAttribute + k]
  792. );
  793. }
  794. }
  795. }
  796. }
  797. /**
  798. * Splits a geometry into multiple geometries, if necessary, to ensure that indices in the
  799. * <code>indices</code> fit into unsigned shorts. This is used to meet the WebGL requirements
  800. * when unsigned int indices are not supported.
  801. * <p>
  802. * If the geometry does not have any <code>indices</code>, this function has no effect.
  803. * </p>
  804. *
  805. * @param {Geometry} geometry The geometry to be split into multiple geometries.
  806. * @returns {Geometry[]} An array of geometries, each with indices that fit into unsigned shorts.
  807. *
  808. * @exception {DeveloperError} geometry.primitiveType must equal to PrimitiveType.TRIANGLES, PrimitiveType.LINES, or PrimitiveType.POINTS
  809. * @exception {DeveloperError} All geometry attribute lists must have the same number of attributes.
  810. *
  811. * @example
  812. * var geometries = Cesium.GeometryPipeline.fitToUnsignedShortIndices(geometry);
  813. */
  814. GeometryPipeline.fitToUnsignedShortIndices = function (geometry) {
  815. //>>includeStart('debug', pragmas.debug);
  816. if (!when.defined(geometry)) {
  817. throw new Check.DeveloperError("geometry is required.");
  818. }
  819. if (
  820. when.defined(geometry.indices) &&
  821. geometry.primitiveType !== GeometryAttribute.PrimitiveType.TRIANGLES &&
  822. geometry.primitiveType !== GeometryAttribute.PrimitiveType.LINES &&
  823. geometry.primitiveType !== GeometryAttribute.PrimitiveType.POINTS
  824. ) {
  825. throw new Check.DeveloperError(
  826. "geometry.primitiveType must equal to PrimitiveType.TRIANGLES, PrimitiveType.LINES, or PrimitiveType.POINTS."
  827. );
  828. }
  829. //>>includeEnd('debug');
  830. var geometries = [];
  831. // If there's an index list and more than 64K attributes, it is possible that
  832. // some indices are outside the range of unsigned short [0, 64K - 1]
  833. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  834. if (
  835. when.defined(geometry.indices) &&
  836. numberOfVertices >= _Math.CesiumMath.SIXTY_FOUR_KILOBYTES
  837. ) {
  838. var oldToNewIndex = [];
  839. var newIndices = [];
  840. var currentIndex = 0;
  841. var newAttributes = copyAttributesDescriptions(geometry.attributes);
  842. var originalIndices = geometry.indices;
  843. var numberOfIndices = originalIndices.length;
  844. var indicesPerPrimitive;
  845. if (geometry.primitiveType === GeometryAttribute.PrimitiveType.TRIANGLES) {
  846. indicesPerPrimitive = 3;
  847. } else if (geometry.primitiveType === GeometryAttribute.PrimitiveType.LINES) {
  848. indicesPerPrimitive = 2;
  849. } else if (geometry.primitiveType === GeometryAttribute.PrimitiveType.POINTS) {
  850. indicesPerPrimitive = 1;
  851. }
  852. for (var j = 0; j < numberOfIndices; j += indicesPerPrimitive) {
  853. for (var k = 0; k < indicesPerPrimitive; ++k) {
  854. var x = originalIndices[j + k];
  855. var i = oldToNewIndex[x];
  856. if (!when.defined(i)) {
  857. i = currentIndex++;
  858. oldToNewIndex[x] = i;
  859. copyVertex(newAttributes, geometry.attributes, x);
  860. }
  861. newIndices.push(i);
  862. }
  863. if (
  864. currentIndex + indicesPerPrimitive >=
  865. _Math.CesiumMath.SIXTY_FOUR_KILOBYTES
  866. ) {
  867. geometries.push(
  868. new GeometryAttribute.Geometry({
  869. attributes: newAttributes,
  870. indices: newIndices,
  871. primitiveType: geometry.primitiveType,
  872. boundingSphere: geometry.boundingSphere,
  873. boundingSphereCV: geometry.boundingSphereCV,
  874. })
  875. );
  876. // Reset for next vertex-array
  877. oldToNewIndex = [];
  878. newIndices = [];
  879. currentIndex = 0;
  880. newAttributes = copyAttributesDescriptions(geometry.attributes);
  881. }
  882. }
  883. if (newIndices.length !== 0) {
  884. geometries.push(
  885. new GeometryAttribute.Geometry({
  886. attributes: newAttributes,
  887. indices: newIndices,
  888. primitiveType: geometry.primitiveType,
  889. boundingSphere: geometry.boundingSphere,
  890. boundingSphereCV: geometry.boundingSphereCV,
  891. })
  892. );
  893. }
  894. } else {
  895. // No need to split into multiple geometries
  896. geometries.push(geometry);
  897. }
  898. return geometries;
  899. };
  900. var scratchProjectTo2DCartesian3 = new Cartesian2.Cartesian3();
  901. var scratchProjectTo2DCartographic = new Cartesian2.Cartographic();
  902. /**
  903. * Projects a geometry's 3D <code>position</code> attribute to 2D, replacing the <code>position</code>
  904. * attribute with separate <code>position3D</code> and <code>position2D</code> attributes.
  905. * <p>
  906. * If the geometry does not have a <code>position</code>, this function has no effect.
  907. * </p>
  908. *
  909. * @param {Geometry} geometry The geometry to modify.
  910. * @param {String} attributeName The name of the attribute.
  911. * @param {String} attributeName3D The name of the attribute in 3D.
  912. * @param {String} attributeName2D The name of the attribute in 2D.
  913. * @param {Object} [projection=new GeographicProjection()] The projection to use.
  914. * @returns {Geometry} The modified <code>geometry</code> argument with <code>position3D</code> and <code>position2D</code> attributes.
  915. *
  916. * @exception {DeveloperError} geometry must have attribute matching the attributeName argument.
  917. * @exception {DeveloperError} The attribute componentDatatype must be ComponentDatatype.DOUBLE.
  918. * @exception {DeveloperError} Could not project a point to 2D.
  919. *
  920. * @example
  921. * geometry = Cesium.GeometryPipeline.projectTo2D(geometry, 'position', 'position3D', 'position2D');
  922. */
  923. GeometryPipeline.projectTo2D = function (
  924. geometry,
  925. attributeName,
  926. attributeName3D,
  927. attributeName2D,
  928. projection
  929. ) {
  930. //>>includeStart('debug', pragmas.debug);
  931. if (!when.defined(geometry)) {
  932. throw new Check.DeveloperError("geometry is required.");
  933. }
  934. if (!when.defined(attributeName)) {
  935. throw new Check.DeveloperError("attributeName is required.");
  936. }
  937. if (!when.defined(attributeName3D)) {
  938. throw new Check.DeveloperError("attributeName3D is required.");
  939. }
  940. if (!when.defined(attributeName2D)) {
  941. throw new Check.DeveloperError("attributeName2D is required.");
  942. }
  943. if (!when.defined(geometry.attributes[attributeName])) {
  944. throw new Check.DeveloperError(
  945. "geometry must have attribute matching the attributeName argument: " +
  946. attributeName +
  947. "."
  948. );
  949. }
  950. if (
  951. geometry.attributes[attributeName].componentDatatype !==
  952. ComponentDatatype.ComponentDatatype.DOUBLE
  953. ) {
  954. throw new Check.DeveloperError(
  955. "The attribute componentDatatype must be ComponentDatatype.DOUBLE."
  956. );
  957. }
  958. //>>includeEnd('debug');
  959. var attribute = geometry.attributes[attributeName];
  960. projection = when.defined(projection) ? projection : new Transforms.GeographicProjection();
  961. var ellipsoid = projection.ellipsoid;
  962. // Project original values to 2D.
  963. var values3D = attribute.values;
  964. var projectedValues = new Float64Array(values3D.length);
  965. var index = 0;
  966. for (var i = 0; i < values3D.length; i += 3) {
  967. var value = Cartesian2.Cartesian3.fromArray(values3D, i, scratchProjectTo2DCartesian3);
  968. var lonLat = ellipsoid.cartesianToCartographic(
  969. value,
  970. scratchProjectTo2DCartographic
  971. );
  972. //>>includeStart('debug', pragmas.debug);
  973. if (!when.defined(lonLat)) {
  974. throw new Check.DeveloperError(
  975. "Could not project point (" +
  976. value.x +
  977. ", " +
  978. value.y +
  979. ", " +
  980. value.z +
  981. ") to 2D."
  982. );
  983. }
  984. //>>includeEnd('debug');
  985. var projectedLonLat = projection.project(
  986. lonLat,
  987. scratchProjectTo2DCartesian3
  988. );
  989. projectedValues[index++] = projectedLonLat.x;
  990. projectedValues[index++] = projectedLonLat.y;
  991. projectedValues[index++] = projectedLonLat.z;
  992. }
  993. // Rename original cartesians to WGS84 cartesians.
  994. geometry.attributes[attributeName3D] = attribute;
  995. // Replace original cartesians with 2D projected cartesians
  996. geometry.attributes[attributeName2D] = new GeometryAttribute.GeometryAttribute({
  997. componentDatatype: ComponentDatatype.ComponentDatatype.DOUBLE,
  998. componentsPerAttribute: 3,
  999. values: projectedValues,
  1000. });
  1001. delete geometry.attributes[attributeName];
  1002. return geometry;
  1003. };
  1004. var encodedResult = {
  1005. high: 0.0,
  1006. low: 0.0,
  1007. };
  1008. /**
  1009. * Encodes floating-point geometry attribute values as two separate attributes to improve
  1010. * rendering precision.
  1011. * <p>
  1012. * This is commonly used to create high-precision position vertex attributes.
  1013. * </p>
  1014. *
  1015. * @param {Geometry} geometry The geometry to modify.
  1016. * @param {String} attributeName The name of the attribute.
  1017. * @param {String} attributeHighName The name of the attribute for the encoded high bits.
  1018. * @param {String} attributeLowName The name of the attribute for the encoded low bits.
  1019. * @returns {Geometry} The modified <code>geometry</code> argument, with its encoded attribute.
  1020. *
  1021. * @exception {DeveloperError} geometry must have attribute matching the attributeName argument.
  1022. * @exception {DeveloperError} The attribute componentDatatype must be ComponentDatatype.DOUBLE.
  1023. *
  1024. * @example
  1025. * geometry = Cesium.GeometryPipeline.encodeAttribute(geometry, 'position3D', 'position3DHigh', 'position3DLow');
  1026. */
  1027. GeometryPipeline.encodeAttribute = function (
  1028. geometry,
  1029. attributeName,
  1030. attributeHighName,
  1031. attributeLowName
  1032. ) {
  1033. //>>includeStart('debug', pragmas.debug);
  1034. if (!when.defined(geometry)) {
  1035. throw new Check.DeveloperError("geometry is required.");
  1036. }
  1037. if (!when.defined(attributeName)) {
  1038. throw new Check.DeveloperError("attributeName is required.");
  1039. }
  1040. if (!when.defined(attributeHighName)) {
  1041. throw new Check.DeveloperError("attributeHighName is required.");
  1042. }
  1043. if (!when.defined(attributeLowName)) {
  1044. throw new Check.DeveloperError("attributeLowName is required.");
  1045. }
  1046. if (!when.defined(geometry.attributes[attributeName])) {
  1047. throw new Check.DeveloperError(
  1048. "geometry must have attribute matching the attributeName argument: " +
  1049. attributeName +
  1050. "."
  1051. );
  1052. }
  1053. if (
  1054. geometry.attributes[attributeName].componentDatatype !==
  1055. ComponentDatatype.ComponentDatatype.DOUBLE
  1056. ) {
  1057. throw new Check.DeveloperError(
  1058. "The attribute componentDatatype must be ComponentDatatype.DOUBLE."
  1059. );
  1060. }
  1061. //>>includeEnd('debug');
  1062. var attribute = geometry.attributes[attributeName];
  1063. var values = attribute.values;
  1064. var length = values.length;
  1065. var highValues = new Float32Array(length);
  1066. var lowValues = new Float32Array(length);
  1067. for (var i = 0; i < length; ++i) {
  1068. EncodedCartesian3.EncodedCartesian3.encode(values[i], encodedResult);
  1069. highValues[i] = encodedResult.high;
  1070. lowValues[i] = encodedResult.low;
  1071. }
  1072. var componentsPerAttribute = attribute.componentsPerAttribute;
  1073. geometry.attributes[attributeHighName] = new GeometryAttribute.GeometryAttribute({
  1074. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1075. componentsPerAttribute: componentsPerAttribute,
  1076. values: highValues,
  1077. });
  1078. geometry.attributes[attributeLowName] = new GeometryAttribute.GeometryAttribute({
  1079. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1080. componentsPerAttribute: componentsPerAttribute,
  1081. values: lowValues,
  1082. });
  1083. delete geometry.attributes[attributeName];
  1084. return geometry;
  1085. };
  1086. var scratchCartesian3$1 = new Cartesian2.Cartesian3();
  1087. function transformPoint(matrix, attribute) {
  1088. if (when.defined(attribute)) {
  1089. var values = attribute.values;
  1090. var length = values.length;
  1091. for (var i = 0; i < length; i += 3) {
  1092. Cartesian2.Cartesian3.unpack(values, i, scratchCartesian3$1);
  1093. Transforms.Matrix4.multiplyByPoint(matrix, scratchCartesian3$1, scratchCartesian3$1);
  1094. Cartesian2.Cartesian3.pack(scratchCartesian3$1, values, i);
  1095. }
  1096. }
  1097. }
  1098. function transformVector(matrix, attribute) {
  1099. if (when.defined(attribute)) {
  1100. var values = attribute.values;
  1101. var length = values.length;
  1102. for (var i = 0; i < length; i += 3) {
  1103. Cartesian2.Cartesian3.unpack(values, i, scratchCartesian3$1);
  1104. Transforms.Matrix3.multiplyByVector(matrix, scratchCartesian3$1, scratchCartesian3$1);
  1105. scratchCartesian3$1 = Cartesian2.Cartesian3.normalize(
  1106. scratchCartesian3$1,
  1107. scratchCartesian3$1
  1108. );
  1109. Cartesian2.Cartesian3.pack(scratchCartesian3$1, values, i);
  1110. }
  1111. }
  1112. }
  1113. var inverseTranspose = new Transforms.Matrix4();
  1114. var normalMatrix = new Transforms.Matrix3();
  1115. /**
  1116. * Transforms a geometry instance to world coordinates. This changes
  1117. * the instance's <code>modelMatrix</code> to {@link Matrix4.IDENTITY} and transforms the
  1118. * following attributes if they are present: <code>position</code>, <code>normal</code>,
  1119. * <code>tangent</code>, and <code>bitangent</code>.
  1120. *
  1121. * @param {GeometryInstance} instance The geometry instance to modify.
  1122. * @returns {GeometryInstance} The modified <code>instance</code> argument, with its attributes transforms to world coordinates.
  1123. *
  1124. * @example
  1125. * Cesium.GeometryPipeline.transformToWorldCoordinates(instance);
  1126. */
  1127. GeometryPipeline.transformToWorldCoordinates = function (instance) {
  1128. //>>includeStart('debug', pragmas.debug);
  1129. if (!when.defined(instance)) {
  1130. throw new Check.DeveloperError("instance is required.");
  1131. }
  1132. //>>includeEnd('debug');
  1133. var modelMatrix = instance.modelMatrix;
  1134. if (Transforms.Matrix4.equals(modelMatrix, Transforms.Matrix4.IDENTITY)) {
  1135. // Already in world coordinates
  1136. return instance;
  1137. }
  1138. var attributes = instance.geometry.attributes;
  1139. // Transform attributes in known vertex formats
  1140. transformPoint(modelMatrix, attributes.position);
  1141. transformPoint(modelMatrix, attributes.prevPosition);
  1142. transformPoint(modelMatrix, attributes.nextPosition);
  1143. if (
  1144. when.defined(attributes.normal) ||
  1145. when.defined(attributes.tangent) ||
  1146. when.defined(attributes.bitangent)
  1147. ) {
  1148. Transforms.Matrix4.inverse(modelMatrix, inverseTranspose);
  1149. Transforms.Matrix4.transpose(inverseTranspose, inverseTranspose);
  1150. Transforms.Matrix4.getMatrix3(inverseTranspose, normalMatrix);
  1151. transformVector(normalMatrix, attributes.normal);
  1152. transformVector(normalMatrix, attributes.tangent);
  1153. transformVector(normalMatrix, attributes.bitangent);
  1154. }
  1155. var boundingSphere = instance.geometry.boundingSphere;
  1156. if (when.defined(boundingSphere)) {
  1157. instance.geometry.boundingSphere = Transforms.BoundingSphere.transform(
  1158. boundingSphere,
  1159. modelMatrix,
  1160. boundingSphere
  1161. );
  1162. }
  1163. instance.modelMatrix = Transforms.Matrix4.clone(Transforms.Matrix4.IDENTITY);
  1164. return instance;
  1165. };
  1166. function findAttributesInAllGeometries(instances, propertyName) {
  1167. var length = instances.length;
  1168. var attributesInAllGeometries = {};
  1169. var attributes0 = instances[0][propertyName].attributes;
  1170. var name;
  1171. for (name in attributes0) {
  1172. if (
  1173. attributes0.hasOwnProperty(name) &&
  1174. when.defined(attributes0[name]) &&
  1175. when.defined(attributes0[name].values)
  1176. ) {
  1177. var attribute = attributes0[name];
  1178. var numberOfComponents = attribute.values.length;
  1179. var inAllGeometries = true;
  1180. // Does this same attribute exist in all geometries?
  1181. for (var i = 1; i < length; ++i) {
  1182. var otherAttribute = instances[i][propertyName].attributes[name];
  1183. if (
  1184. !when.defined(otherAttribute) ||
  1185. attribute.componentDatatype !== otherAttribute.componentDatatype ||
  1186. attribute.componentsPerAttribute !==
  1187. otherAttribute.componentsPerAttribute ||
  1188. attribute.normalize !== otherAttribute.normalize
  1189. ) {
  1190. inAllGeometries = false;
  1191. break;
  1192. }
  1193. numberOfComponents += otherAttribute.values.length;
  1194. }
  1195. if (inAllGeometries) {
  1196. attributesInAllGeometries[name] = new GeometryAttribute.GeometryAttribute({
  1197. componentDatatype: attribute.componentDatatype,
  1198. componentsPerAttribute: attribute.componentsPerAttribute,
  1199. normalize: attribute.normalize,
  1200. values: ComponentDatatype.ComponentDatatype.createTypedArray(
  1201. attribute.componentDatatype,
  1202. numberOfComponents
  1203. ),
  1204. });
  1205. }
  1206. }
  1207. }
  1208. return attributesInAllGeometries;
  1209. }
  1210. var tempScratch = new Cartesian2.Cartesian3();
  1211. function combineGeometries(instances, propertyName) {
  1212. var length = instances.length;
  1213. var name;
  1214. var i;
  1215. var j;
  1216. var k;
  1217. var m = instances[0].modelMatrix;
  1218. var haveIndices = when.defined(instances[0][propertyName].indices);
  1219. var primitiveType = instances[0][propertyName].primitiveType;
  1220. //>>includeStart('debug', pragmas.debug);
  1221. for (i = 1; i < length; ++i) {
  1222. if (!Transforms.Matrix4.equals(instances[i].modelMatrix, m)) {
  1223. throw new Check.DeveloperError("All instances must have the same modelMatrix.");
  1224. }
  1225. if (when.defined(instances[i][propertyName].indices) !== haveIndices) {
  1226. throw new Check.DeveloperError(
  1227. "All instance geometries must have an indices or not have one."
  1228. );
  1229. }
  1230. if (instances[i][propertyName].primitiveType !== primitiveType) {
  1231. throw new Check.DeveloperError(
  1232. "All instance geometries must have the same primitiveType."
  1233. );
  1234. }
  1235. }
  1236. //>>includeEnd('debug');
  1237. // Find subset of attributes in all geometries
  1238. var attributes = findAttributesInAllGeometries(instances, propertyName);
  1239. var values;
  1240. var sourceValues;
  1241. var sourceValuesLength;
  1242. // Combine attributes from each geometry into a single typed array
  1243. for (name in attributes) {
  1244. if (attributes.hasOwnProperty(name)) {
  1245. values = attributes[name].values;
  1246. k = 0;
  1247. for (i = 0; i < length; ++i) {
  1248. sourceValues = instances[i][propertyName].attributes[name].values;
  1249. sourceValuesLength = sourceValues.length;
  1250. for (j = 0; j < sourceValuesLength; ++j) {
  1251. values[k++] = sourceValues[j];
  1252. }
  1253. }
  1254. }
  1255. }
  1256. // Combine index lists
  1257. var indices;
  1258. if (haveIndices) {
  1259. var numberOfIndices = 0;
  1260. for (i = 0; i < length; ++i) {
  1261. numberOfIndices += instances[i][propertyName].indices.length;
  1262. }
  1263. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(
  1264. new GeometryAttribute.Geometry({
  1265. attributes: attributes,
  1266. primitiveType: GeometryAttribute.PrimitiveType.POINTS,
  1267. })
  1268. );
  1269. var destIndices = IndexDatatype.IndexDatatype.createTypedArray(
  1270. numberOfVertices,
  1271. numberOfIndices
  1272. );
  1273. var destOffset = 0;
  1274. var offset = 0;
  1275. for (i = 0; i < length; ++i) {
  1276. var sourceIndices = instances[i][propertyName].indices;
  1277. var sourceIndicesLen = sourceIndices.length;
  1278. for (k = 0; k < sourceIndicesLen; ++k) {
  1279. destIndices[destOffset++] = offset + sourceIndices[k];
  1280. }
  1281. offset += GeometryAttribute.Geometry.computeNumberOfVertices(instances[i][propertyName]);
  1282. }
  1283. indices = destIndices;
  1284. }
  1285. // Create bounding sphere that includes all instances
  1286. var center = new Cartesian2.Cartesian3();
  1287. var radius = 0.0;
  1288. var bs;
  1289. for (i = 0; i < length; ++i) {
  1290. bs = instances[i][propertyName].boundingSphere;
  1291. if (!when.defined(bs)) {
  1292. // If any geometries have an undefined bounding sphere, then so does the combined geometry
  1293. center = undefined;
  1294. break;
  1295. }
  1296. Cartesian2.Cartesian3.add(bs.center, center, center);
  1297. }
  1298. if (when.defined(center)) {
  1299. Cartesian2.Cartesian3.divideByScalar(center, length, center);
  1300. for (i = 0; i < length; ++i) {
  1301. bs = instances[i][propertyName].boundingSphere;
  1302. var tempRadius =
  1303. Cartesian2.Cartesian3.magnitude(
  1304. Cartesian2.Cartesian3.subtract(bs.center, center, tempScratch)
  1305. ) + bs.radius;
  1306. if (tempRadius > radius) {
  1307. radius = tempRadius;
  1308. }
  1309. }
  1310. }
  1311. return new GeometryAttribute.Geometry({
  1312. attributes: attributes,
  1313. indices: indices,
  1314. primitiveType: primitiveType,
  1315. boundingSphere: when.defined(center)
  1316. ? new Transforms.BoundingSphere(center, radius)
  1317. : undefined,
  1318. });
  1319. }
  1320. /**
  1321. * Combines geometry from several {@link GeometryInstance} objects into one geometry.
  1322. * This concatenates the attributes, concatenates and adjusts the indices, and creates
  1323. * a bounding sphere encompassing all instances.
  1324. * <p>
  1325. * If the instances do not have the same attributes, a subset of attributes common
  1326. * to all instances is used, and the others are ignored.
  1327. * </p>
  1328. * <p>
  1329. * This is used by {@link Primitive} to efficiently render a large amount of static data.
  1330. * </p>
  1331. *
  1332. * @private
  1333. *
  1334. * @param {GeometryInstance[]} [instances] The array of {@link GeometryInstance} objects whose geometry will be combined.
  1335. * @returns {Geometry} A single geometry created from the provided geometry instances.
  1336. *
  1337. * @exception {DeveloperError} All instances must have the same modelMatrix.
  1338. * @exception {DeveloperError} All instance geometries must have an indices or not have one.
  1339. * @exception {DeveloperError} All instance geometries must have the same primitiveType.
  1340. *
  1341. *
  1342. * @example
  1343. * for (var i = 0; i < instances.length; ++i) {
  1344. * Cesium.GeometryPipeline.transformToWorldCoordinates(instances[i]);
  1345. * }
  1346. * var geometries = Cesium.GeometryPipeline.combineInstances(instances);
  1347. *
  1348. * @see GeometryPipeline.transformToWorldCoordinates
  1349. */
  1350. GeometryPipeline.combineInstances = function (instances) {
  1351. //>>includeStart('debug', pragmas.debug);
  1352. if (!when.defined(instances) || instances.length < 1) {
  1353. throw new Check.DeveloperError(
  1354. "instances is required and must have length greater than zero."
  1355. );
  1356. }
  1357. //>>includeEnd('debug');
  1358. var instanceGeometry = [];
  1359. var instanceSplitGeometry = [];
  1360. var length = instances.length;
  1361. for (var i = 0; i < length; ++i) {
  1362. var instance = instances[i];
  1363. if (when.defined(instance.geometry)) {
  1364. instanceGeometry.push(instance);
  1365. } else if (
  1366. when.defined(instance.westHemisphereGeometry) &&
  1367. when.defined(instance.eastHemisphereGeometry)
  1368. ) {
  1369. instanceSplitGeometry.push(instance);
  1370. }
  1371. }
  1372. var geometries = [];
  1373. if (instanceGeometry.length > 0) {
  1374. geometries.push(combineGeometries(instanceGeometry, "geometry"));
  1375. }
  1376. if (instanceSplitGeometry.length > 0) {
  1377. geometries.push(
  1378. combineGeometries(instanceSplitGeometry, "westHemisphereGeometry")
  1379. );
  1380. geometries.push(
  1381. combineGeometries(instanceSplitGeometry, "eastHemisphereGeometry")
  1382. );
  1383. }
  1384. return geometries;
  1385. };
  1386. var normal = new Cartesian2.Cartesian3();
  1387. var v0 = new Cartesian2.Cartesian3();
  1388. var v1 = new Cartesian2.Cartesian3();
  1389. var v2 = new Cartesian2.Cartesian3();
  1390. /**
  1391. * Computes per-vertex normals for a geometry containing <code>TRIANGLES</code> by averaging the normals of
  1392. * all triangles incident to the vertex. The result is a new <code>normal</code> attribute added to the geometry.
  1393. * This assumes a counter-clockwise winding order.
  1394. *
  1395. * @param {Geometry} geometry The geometry to modify.
  1396. * @returns {Geometry} The modified <code>geometry</code> argument with the computed <code>normal</code> attribute.
  1397. *
  1398. * @exception {DeveloperError} geometry.indices length must be greater than 0 and be a multiple of 3.
  1399. * @exception {DeveloperError} geometry.primitiveType must be {@link PrimitiveType.TRIANGLES}.
  1400. *
  1401. * @example
  1402. * Cesium.GeometryPipeline.computeNormal(geometry);
  1403. */
  1404. GeometryPipeline.computeNormal = function (geometry) {
  1405. //>>includeStart('debug', pragmas.debug);
  1406. if (!when.defined(geometry)) {
  1407. throw new Check.DeveloperError("geometry is required.");
  1408. }
  1409. if (
  1410. !when.defined(geometry.attributes.position) ||
  1411. !when.defined(geometry.attributes.position.values)
  1412. ) {
  1413. throw new Check.DeveloperError(
  1414. "geometry.attributes.position.values is required."
  1415. );
  1416. }
  1417. if (!when.defined(geometry.indices)) {
  1418. throw new Check.DeveloperError("geometry.indices is required.");
  1419. }
  1420. if (geometry.indices.length < 2 || geometry.indices.length % 3 !== 0) {
  1421. throw new Check.DeveloperError(
  1422. "geometry.indices length must be greater than 0 and be a multiple of 3."
  1423. );
  1424. }
  1425. if (geometry.primitiveType !== GeometryAttribute.PrimitiveType.TRIANGLES) {
  1426. throw new Check.DeveloperError(
  1427. "geometry.primitiveType must be PrimitiveType.TRIANGLES."
  1428. );
  1429. }
  1430. //>>includeEnd('debug');
  1431. var indices = geometry.indices;
  1432. var attributes = geometry.attributes;
  1433. var vertices = attributes.position.values;
  1434. var numVertices = attributes.position.values.length / 3;
  1435. var numIndices = indices.length;
  1436. var normalsPerVertex = new Array(numVertices);
  1437. var normalsPerTriangle = new Array(numIndices / 3);
  1438. var normalIndices = new Array(numIndices);
  1439. var i;
  1440. for (i = 0; i < numVertices; i++) {
  1441. normalsPerVertex[i] = {
  1442. indexOffset: 0,
  1443. count: 0,
  1444. currentCount: 0,
  1445. };
  1446. }
  1447. var j = 0;
  1448. for (i = 0; i < numIndices; i += 3) {
  1449. var i0 = indices[i];
  1450. var i1 = indices[i + 1];
  1451. var i2 = indices[i + 2];
  1452. var i03 = i0 * 3;
  1453. var i13 = i1 * 3;
  1454. var i23 = i2 * 3;
  1455. v0.x = vertices[i03];
  1456. v0.y = vertices[i03 + 1];
  1457. v0.z = vertices[i03 + 2];
  1458. v1.x = vertices[i13];
  1459. v1.y = vertices[i13 + 1];
  1460. v1.z = vertices[i13 + 2];
  1461. v2.x = vertices[i23];
  1462. v2.y = vertices[i23 + 1];
  1463. v2.z = vertices[i23 + 2];
  1464. normalsPerVertex[i0].count++;
  1465. normalsPerVertex[i1].count++;
  1466. normalsPerVertex[i2].count++;
  1467. Cartesian2.Cartesian3.subtract(v1, v0, v1);
  1468. Cartesian2.Cartesian3.subtract(v2, v0, v2);
  1469. normalsPerTriangle[j] = Cartesian2.Cartesian3.cross(v1, v2, new Cartesian2.Cartesian3());
  1470. j++;
  1471. }
  1472. var indexOffset = 0;
  1473. for (i = 0; i < numVertices; i++) {
  1474. normalsPerVertex[i].indexOffset += indexOffset;
  1475. indexOffset += normalsPerVertex[i].count;
  1476. }
  1477. j = 0;
  1478. var vertexNormalData;
  1479. for (i = 0; i < numIndices; i += 3) {
  1480. vertexNormalData = normalsPerVertex[indices[i]];
  1481. var index = vertexNormalData.indexOffset + vertexNormalData.currentCount;
  1482. normalIndices[index] = j;
  1483. vertexNormalData.currentCount++;
  1484. vertexNormalData = normalsPerVertex[indices[i + 1]];
  1485. index = vertexNormalData.indexOffset + vertexNormalData.currentCount;
  1486. normalIndices[index] = j;
  1487. vertexNormalData.currentCount++;
  1488. vertexNormalData = normalsPerVertex[indices[i + 2]];
  1489. index = vertexNormalData.indexOffset + vertexNormalData.currentCount;
  1490. normalIndices[index] = j;
  1491. vertexNormalData.currentCount++;
  1492. j++;
  1493. }
  1494. var normalValues = new Float32Array(numVertices * 3);
  1495. for (i = 0; i < numVertices; i++) {
  1496. var i3 = i * 3;
  1497. vertexNormalData = normalsPerVertex[i];
  1498. Cartesian2.Cartesian3.clone(Cartesian2.Cartesian3.ZERO, normal);
  1499. if (vertexNormalData.count > 0) {
  1500. for (j = 0; j < vertexNormalData.count; j++) {
  1501. Cartesian2.Cartesian3.add(
  1502. normal,
  1503. normalsPerTriangle[normalIndices[vertexNormalData.indexOffset + j]],
  1504. normal
  1505. );
  1506. }
  1507. // We can run into an issue where a vertex is used with 2 primitives that have opposite winding order.
  1508. if (
  1509. Cartesian2.Cartesian3.equalsEpsilon(Cartesian2.Cartesian3.ZERO, normal, _Math.CesiumMath.EPSILON10)
  1510. ) {
  1511. Cartesian2.Cartesian3.clone(
  1512. normalsPerTriangle[normalIndices[vertexNormalData.indexOffset]],
  1513. normal
  1514. );
  1515. }
  1516. }
  1517. // We end up with a zero vector probably because of a degenerate triangle
  1518. if (
  1519. Cartesian2.Cartesian3.equalsEpsilon(Cartesian2.Cartesian3.ZERO, normal, _Math.CesiumMath.EPSILON10)
  1520. ) {
  1521. // Default to (0,0,1)
  1522. normal.z = 1.0;
  1523. }
  1524. Cartesian2.Cartesian3.normalize(normal, normal);
  1525. normalValues[i3] = normal.x;
  1526. normalValues[i3 + 1] = normal.y;
  1527. normalValues[i3 + 2] = normal.z;
  1528. }
  1529. geometry.attributes.normal = new GeometryAttribute.GeometryAttribute({
  1530. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1531. componentsPerAttribute: 3,
  1532. values: normalValues,
  1533. });
  1534. return geometry;
  1535. };
  1536. var normalScratch = new Cartesian2.Cartesian3();
  1537. var normalScale = new Cartesian2.Cartesian3();
  1538. var tScratch = new Cartesian2.Cartesian3();
  1539. /**
  1540. * Computes per-vertex tangents and bitangents for a geometry containing <code>TRIANGLES</code>.
  1541. * The result is new <code>tangent</code> and <code>bitangent</code> attributes added to the geometry.
  1542. * This assumes a counter-clockwise winding order.
  1543. * <p>
  1544. * Based on <a href="http://www.terathon.com/code/tangent.html">Computing Tangent Space Basis Vectors
  1545. * for an Arbitrary Mesh</a> by Eric Lengyel.
  1546. * </p>
  1547. *
  1548. * @param {Geometry} geometry The geometry to modify.
  1549. * @returns {Geometry} The modified <code>geometry</code> argument with the computed <code>tangent</code> and <code>bitangent</code> attributes.
  1550. *
  1551. * @exception {DeveloperError} geometry.indices length must be greater than 0 and be a multiple of 3.
  1552. * @exception {DeveloperError} geometry.primitiveType must be {@link PrimitiveType.TRIANGLES}.
  1553. *
  1554. * @example
  1555. * Cesium.GeometryPipeline.computeTangentAndBiTangent(geometry);
  1556. */
  1557. GeometryPipeline.computeTangentAndBitangent = function (geometry) {
  1558. //>>includeStart('debug', pragmas.debug);
  1559. if (!when.defined(geometry)) {
  1560. throw new Check.DeveloperError("geometry is required.");
  1561. }
  1562. //>>includeEnd('debug');
  1563. var attributes = geometry.attributes;
  1564. var indices = geometry.indices;
  1565. //>>includeStart('debug', pragmas.debug);
  1566. if (!when.defined(attributes.position) || !when.defined(attributes.position.values)) {
  1567. throw new Check.DeveloperError(
  1568. "geometry.attributes.position.values is required."
  1569. );
  1570. }
  1571. if (!when.defined(attributes.normal) || !when.defined(attributes.normal.values)) {
  1572. throw new Check.DeveloperError("geometry.attributes.normal.values is required.");
  1573. }
  1574. if (!when.defined(attributes.st) || !when.defined(attributes.st.values)) {
  1575. throw new Check.DeveloperError("geometry.attributes.st.values is required.");
  1576. }
  1577. if (!when.defined(indices)) {
  1578. throw new Check.DeveloperError("geometry.indices is required.");
  1579. }
  1580. if (indices.length < 2 || indices.length % 3 !== 0) {
  1581. throw new Check.DeveloperError(
  1582. "geometry.indices length must be greater than 0 and be a multiple of 3."
  1583. );
  1584. }
  1585. if (geometry.primitiveType !== GeometryAttribute.PrimitiveType.TRIANGLES) {
  1586. throw new Check.DeveloperError(
  1587. "geometry.primitiveType must be PrimitiveType.TRIANGLES."
  1588. );
  1589. }
  1590. //>>includeEnd('debug');
  1591. var vertices = geometry.attributes.position.values;
  1592. var normals = geometry.attributes.normal.values;
  1593. var st = geometry.attributes.st.values;
  1594. var numVertices = geometry.attributes.position.values.length / 3;
  1595. var numIndices = indices.length;
  1596. var tan1 = new Array(numVertices * 3);
  1597. var i;
  1598. for (i = 0; i < tan1.length; i++) {
  1599. tan1[i] = 0;
  1600. }
  1601. var i03;
  1602. var i13;
  1603. var i23;
  1604. for (i = 0; i < numIndices; i += 3) {
  1605. var i0 = indices[i];
  1606. var i1 = indices[i + 1];
  1607. var i2 = indices[i + 2];
  1608. i03 = i0 * 3;
  1609. i13 = i1 * 3;
  1610. i23 = i2 * 3;
  1611. var i02 = i0 * 2;
  1612. var i12 = i1 * 2;
  1613. var i22 = i2 * 2;
  1614. var ux = vertices[i03];
  1615. var uy = vertices[i03 + 1];
  1616. var uz = vertices[i03 + 2];
  1617. var wx = st[i02];
  1618. var wy = st[i02 + 1];
  1619. var t1 = st[i12 + 1] - wy;
  1620. var t2 = st[i22 + 1] - wy;
  1621. var r = 1.0 / ((st[i12] - wx) * t2 - (st[i22] - wx) * t1);
  1622. var sdirx = (t2 * (vertices[i13] - ux) - t1 * (vertices[i23] - ux)) * r;
  1623. var sdiry =
  1624. (t2 * (vertices[i13 + 1] - uy) - t1 * (vertices[i23 + 1] - uy)) * r;
  1625. var sdirz =
  1626. (t2 * (vertices[i13 + 2] - uz) - t1 * (vertices[i23 + 2] - uz)) * r;
  1627. tan1[i03] += sdirx;
  1628. tan1[i03 + 1] += sdiry;
  1629. tan1[i03 + 2] += sdirz;
  1630. tan1[i13] += sdirx;
  1631. tan1[i13 + 1] += sdiry;
  1632. tan1[i13 + 2] += sdirz;
  1633. tan1[i23] += sdirx;
  1634. tan1[i23 + 1] += sdiry;
  1635. tan1[i23 + 2] += sdirz;
  1636. }
  1637. var tangentValues = new Float32Array(numVertices * 3);
  1638. var bitangentValues = new Float32Array(numVertices * 3);
  1639. for (i = 0; i < numVertices; i++) {
  1640. i03 = i * 3;
  1641. i13 = i03 + 1;
  1642. i23 = i03 + 2;
  1643. var n = Cartesian2.Cartesian3.fromArray(normals, i03, normalScratch);
  1644. var t = Cartesian2.Cartesian3.fromArray(tan1, i03, tScratch);
  1645. var scalar = Cartesian2.Cartesian3.dot(n, t);
  1646. Cartesian2.Cartesian3.multiplyByScalar(n, scalar, normalScale);
  1647. Cartesian2.Cartesian3.normalize(Cartesian2.Cartesian3.subtract(t, normalScale, t), t);
  1648. tangentValues[i03] = t.x;
  1649. tangentValues[i13] = t.y;
  1650. tangentValues[i23] = t.z;
  1651. Cartesian2.Cartesian3.normalize(Cartesian2.Cartesian3.cross(n, t, t), t);
  1652. bitangentValues[i03] = t.x;
  1653. bitangentValues[i13] = t.y;
  1654. bitangentValues[i23] = t.z;
  1655. }
  1656. geometry.attributes.tangent = new GeometryAttribute.GeometryAttribute({
  1657. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1658. componentsPerAttribute: 3,
  1659. values: tangentValues,
  1660. });
  1661. geometry.attributes.bitangent = new GeometryAttribute.GeometryAttribute({
  1662. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1663. componentsPerAttribute: 3,
  1664. values: bitangentValues,
  1665. });
  1666. return geometry;
  1667. };
  1668. var scratchCartesian2$1 = new Cartesian2.Cartesian2();
  1669. var toEncode1 = new Cartesian2.Cartesian3();
  1670. var toEncode2 = new Cartesian2.Cartesian3();
  1671. var toEncode3 = new Cartesian2.Cartesian3();
  1672. var encodeResult2 = new Cartesian2.Cartesian2();
  1673. /**
  1674. * Compresses and packs geometry normal attribute values to save memory.
  1675. *
  1676. * @param {Geometry} geometry The geometry to modify.
  1677. * @returns {Geometry} The modified <code>geometry</code> argument, with its normals compressed and packed.
  1678. *
  1679. * @example
  1680. * geometry = Cesium.GeometryPipeline.compressVertices(geometry);
  1681. */
  1682. GeometryPipeline.compressVertices = function (geometry) {
  1683. //>>includeStart('debug', pragmas.debug);
  1684. if (!when.defined(geometry)) {
  1685. throw new Check.DeveloperError("geometry is required.");
  1686. }
  1687. //>>includeEnd('debug');
  1688. var extrudeAttribute = geometry.attributes.extrudeDirection;
  1689. var i;
  1690. var numVertices;
  1691. if (when.defined(extrudeAttribute)) {
  1692. //only shadow volumes use extrudeDirection, and shadow volumes use vertexFormat: POSITION_ONLY so we don't need to check other attributes
  1693. var extrudeDirections = extrudeAttribute.values;
  1694. numVertices = extrudeDirections.length / 3.0;
  1695. var compressedDirections = new Float32Array(numVertices * 2);
  1696. var i2 = 0;
  1697. for (i = 0; i < numVertices; ++i) {
  1698. Cartesian2.Cartesian3.fromArray(extrudeDirections, i * 3.0, toEncode1);
  1699. if (Cartesian2.Cartesian3.equals(toEncode1, Cartesian2.Cartesian3.ZERO)) {
  1700. i2 += 2;
  1701. continue;
  1702. }
  1703. encodeResult2 = AttributeCompression.AttributeCompression.octEncodeInRange(
  1704. toEncode1,
  1705. 65535,
  1706. encodeResult2
  1707. );
  1708. compressedDirections[i2++] = encodeResult2.x;
  1709. compressedDirections[i2++] = encodeResult2.y;
  1710. }
  1711. geometry.attributes.compressedAttributes = new GeometryAttribute.GeometryAttribute({
  1712. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1713. componentsPerAttribute: 2,
  1714. values: compressedDirections,
  1715. });
  1716. delete geometry.attributes.extrudeDirection;
  1717. return geometry;
  1718. }
  1719. var normalAttribute = geometry.attributes.normal;
  1720. var stAttribute = geometry.attributes.st;
  1721. var hasNormal = when.defined(normalAttribute);
  1722. var hasSt = when.defined(stAttribute);
  1723. if (!hasNormal && !hasSt) {
  1724. return geometry;
  1725. }
  1726. var tangentAttribute = geometry.attributes.tangent;
  1727. var bitangentAttribute = geometry.attributes.bitangent;
  1728. var hasTangent = when.defined(tangentAttribute);
  1729. var hasBitangent = when.defined(bitangentAttribute);
  1730. var normals;
  1731. var st;
  1732. var tangents;
  1733. var bitangents;
  1734. if (hasNormal) {
  1735. normals = normalAttribute.values;
  1736. }
  1737. if (hasSt) {
  1738. st = stAttribute.values;
  1739. }
  1740. if (hasTangent) {
  1741. tangents = tangentAttribute.values;
  1742. }
  1743. if (hasBitangent) {
  1744. bitangents = bitangentAttribute.values;
  1745. }
  1746. var length = hasNormal ? normals.length : st.length;
  1747. var numComponents = hasNormal ? 3.0 : 2.0;
  1748. numVertices = length / numComponents;
  1749. var compressedLength = numVertices;
  1750. var numCompressedComponents = hasSt && hasNormal ? 2.0 : 1.0;
  1751. numCompressedComponents += hasTangent || hasBitangent ? 1.0 : 0.0;
  1752. compressedLength *= numCompressedComponents;
  1753. var compressedAttributes = new Float32Array(compressedLength);
  1754. var normalIndex = 0;
  1755. for (i = 0; i < numVertices; ++i) {
  1756. if (hasSt) {
  1757. Cartesian2.Cartesian2.fromArray(st, i * 2.0, scratchCartesian2$1);
  1758. compressedAttributes[
  1759. normalIndex++
  1760. ] = AttributeCompression.AttributeCompression.compressTextureCoordinates(scratchCartesian2$1);
  1761. }
  1762. var index = i * 3.0;
  1763. if (hasNormal && when.defined(tangents) && when.defined(bitangents)) {
  1764. Cartesian2.Cartesian3.fromArray(normals, index, toEncode1);
  1765. Cartesian2.Cartesian3.fromArray(tangents, index, toEncode2);
  1766. Cartesian2.Cartesian3.fromArray(bitangents, index, toEncode3);
  1767. AttributeCompression.AttributeCompression.octPack(
  1768. toEncode1,
  1769. toEncode2,
  1770. toEncode3,
  1771. scratchCartesian2$1
  1772. );
  1773. compressedAttributes[normalIndex++] = scratchCartesian2$1.x;
  1774. compressedAttributes[normalIndex++] = scratchCartesian2$1.y;
  1775. } else {
  1776. if (hasNormal) {
  1777. Cartesian2.Cartesian3.fromArray(normals, index, toEncode1);
  1778. compressedAttributes[
  1779. normalIndex++
  1780. ] = AttributeCompression.AttributeCompression.octEncodeFloat(toEncode1);
  1781. }
  1782. if (hasTangent) {
  1783. Cartesian2.Cartesian3.fromArray(tangents, index, toEncode1);
  1784. compressedAttributes[
  1785. normalIndex++
  1786. ] = AttributeCompression.AttributeCompression.octEncodeFloat(toEncode1);
  1787. }
  1788. if (hasBitangent) {
  1789. Cartesian2.Cartesian3.fromArray(bitangents, index, toEncode1);
  1790. compressedAttributes[
  1791. normalIndex++
  1792. ] = AttributeCompression.AttributeCompression.octEncodeFloat(toEncode1);
  1793. }
  1794. }
  1795. }
  1796. geometry.attributes.compressedAttributes = new GeometryAttribute.GeometryAttribute({
  1797. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  1798. componentsPerAttribute: numCompressedComponents,
  1799. values: compressedAttributes,
  1800. });
  1801. if (hasNormal) {
  1802. delete geometry.attributes.normal;
  1803. }
  1804. if (hasSt) {
  1805. delete geometry.attributes.st;
  1806. }
  1807. if (hasBitangent) {
  1808. delete geometry.attributes.bitangent;
  1809. }
  1810. if (hasTangent) {
  1811. delete geometry.attributes.tangent;
  1812. }
  1813. return geometry;
  1814. };
  1815. function indexTriangles(geometry) {
  1816. if (when.defined(geometry.indices)) {
  1817. return geometry;
  1818. }
  1819. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  1820. //>>includeStart('debug', pragmas.debug);
  1821. if (numberOfVertices < 3) {
  1822. throw new Check.DeveloperError("The number of vertices must be at least three.");
  1823. }
  1824. if (numberOfVertices % 3 !== 0) {
  1825. throw new Check.DeveloperError(
  1826. "The number of vertices must be a multiple of three."
  1827. );
  1828. }
  1829. //>>includeEnd('debug');
  1830. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  1831. numberOfVertices,
  1832. numberOfVertices
  1833. );
  1834. for (var i = 0; i < numberOfVertices; ++i) {
  1835. indices[i] = i;
  1836. }
  1837. geometry.indices = indices;
  1838. return geometry;
  1839. }
  1840. function indexTriangleFan(geometry) {
  1841. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  1842. //>>includeStart('debug', pragmas.debug);
  1843. if (numberOfVertices < 3) {
  1844. throw new Check.DeveloperError("The number of vertices must be at least three.");
  1845. }
  1846. //>>includeEnd('debug');
  1847. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  1848. numberOfVertices,
  1849. (numberOfVertices - 2) * 3
  1850. );
  1851. indices[0] = 1;
  1852. indices[1] = 0;
  1853. indices[2] = 2;
  1854. var indicesIndex = 3;
  1855. for (var i = 3; i < numberOfVertices; ++i) {
  1856. indices[indicesIndex++] = i - 1;
  1857. indices[indicesIndex++] = 0;
  1858. indices[indicesIndex++] = i;
  1859. }
  1860. geometry.indices = indices;
  1861. geometry.primitiveType = GeometryAttribute.PrimitiveType.TRIANGLES;
  1862. return geometry;
  1863. }
  1864. function indexTriangleStrip(geometry) {
  1865. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  1866. //>>includeStart('debug', pragmas.debug);
  1867. if (numberOfVertices < 3) {
  1868. throw new Check.DeveloperError("The number of vertices must be at least 3.");
  1869. }
  1870. //>>includeEnd('debug');
  1871. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  1872. numberOfVertices,
  1873. (numberOfVertices - 2) * 3
  1874. );
  1875. indices[0] = 0;
  1876. indices[1] = 1;
  1877. indices[2] = 2;
  1878. if (numberOfVertices > 3) {
  1879. indices[3] = 0;
  1880. indices[4] = 2;
  1881. indices[5] = 3;
  1882. }
  1883. var indicesIndex = 6;
  1884. for (var i = 3; i < numberOfVertices - 1; i += 2) {
  1885. indices[indicesIndex++] = i;
  1886. indices[indicesIndex++] = i - 1;
  1887. indices[indicesIndex++] = i + 1;
  1888. if (i + 2 < numberOfVertices) {
  1889. indices[indicesIndex++] = i;
  1890. indices[indicesIndex++] = i + 1;
  1891. indices[indicesIndex++] = i + 2;
  1892. }
  1893. }
  1894. geometry.indices = indices;
  1895. geometry.primitiveType = GeometryAttribute.PrimitiveType.TRIANGLES;
  1896. return geometry;
  1897. }
  1898. function indexLines(geometry) {
  1899. if (when.defined(geometry.indices)) {
  1900. return geometry;
  1901. }
  1902. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  1903. //>>includeStart('debug', pragmas.debug);
  1904. if (numberOfVertices < 2) {
  1905. throw new Check.DeveloperError("The number of vertices must be at least two.");
  1906. }
  1907. if (numberOfVertices % 2 !== 0) {
  1908. throw new Check.DeveloperError("The number of vertices must be a multiple of 2.");
  1909. }
  1910. //>>includeEnd('debug');
  1911. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  1912. numberOfVertices,
  1913. numberOfVertices
  1914. );
  1915. for (var i = 0; i < numberOfVertices; ++i) {
  1916. indices[i] = i;
  1917. }
  1918. geometry.indices = indices;
  1919. return geometry;
  1920. }
  1921. function indexLineStrip(geometry) {
  1922. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  1923. //>>includeStart('debug', pragmas.debug);
  1924. if (numberOfVertices < 2) {
  1925. throw new Check.DeveloperError("The number of vertices must be at least two.");
  1926. }
  1927. //>>includeEnd('debug');
  1928. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  1929. numberOfVertices,
  1930. (numberOfVertices - 1) * 2
  1931. );
  1932. indices[0] = 0;
  1933. indices[1] = 1;
  1934. var indicesIndex = 2;
  1935. for (var i = 2; i < numberOfVertices; ++i) {
  1936. indices[indicesIndex++] = i - 1;
  1937. indices[indicesIndex++] = i;
  1938. }
  1939. geometry.indices = indices;
  1940. geometry.primitiveType = GeometryAttribute.PrimitiveType.LINES;
  1941. return geometry;
  1942. }
  1943. function indexLineLoop(geometry) {
  1944. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  1945. //>>includeStart('debug', pragmas.debug);
  1946. if (numberOfVertices < 2) {
  1947. throw new Check.DeveloperError("The number of vertices must be at least two.");
  1948. }
  1949. //>>includeEnd('debug');
  1950. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  1951. numberOfVertices,
  1952. numberOfVertices * 2
  1953. );
  1954. indices[0] = 0;
  1955. indices[1] = 1;
  1956. var indicesIndex = 2;
  1957. for (var i = 2; i < numberOfVertices; ++i) {
  1958. indices[indicesIndex++] = i - 1;
  1959. indices[indicesIndex++] = i;
  1960. }
  1961. indices[indicesIndex++] = numberOfVertices - 1;
  1962. indices[indicesIndex] = 0;
  1963. geometry.indices = indices;
  1964. geometry.primitiveType = GeometryAttribute.PrimitiveType.LINES;
  1965. return geometry;
  1966. }
  1967. function indexPrimitive(geometry) {
  1968. switch (geometry.primitiveType) {
  1969. case GeometryAttribute.PrimitiveType.TRIANGLE_FAN:
  1970. return indexTriangleFan(geometry);
  1971. case GeometryAttribute.PrimitiveType.TRIANGLE_STRIP:
  1972. return indexTriangleStrip(geometry);
  1973. case GeometryAttribute.PrimitiveType.TRIANGLES:
  1974. return indexTriangles(geometry);
  1975. case GeometryAttribute.PrimitiveType.LINE_STRIP:
  1976. return indexLineStrip(geometry);
  1977. case GeometryAttribute.PrimitiveType.LINE_LOOP:
  1978. return indexLineLoop(geometry);
  1979. case GeometryAttribute.PrimitiveType.LINES:
  1980. return indexLines(geometry);
  1981. }
  1982. return geometry;
  1983. }
  1984. function offsetPointFromXZPlane(p, isBehind) {
  1985. if (Math.abs(p.y) < _Math.CesiumMath.EPSILON6) {
  1986. if (isBehind) {
  1987. p.y = -_Math.CesiumMath.EPSILON6;
  1988. } else {
  1989. p.y = _Math.CesiumMath.EPSILON6;
  1990. }
  1991. }
  1992. }
  1993. function offsetTriangleFromXZPlane(p0, p1, p2) {
  1994. if (p0.y !== 0.0 && p1.y !== 0.0 && p2.y !== 0.0) {
  1995. offsetPointFromXZPlane(p0, p0.y < 0.0);
  1996. offsetPointFromXZPlane(p1, p1.y < 0.0);
  1997. offsetPointFromXZPlane(p2, p2.y < 0.0);
  1998. return;
  1999. }
  2000. var p0y = Math.abs(p0.y);
  2001. var p1y = Math.abs(p1.y);
  2002. var p2y = Math.abs(p2.y);
  2003. var sign;
  2004. if (p0y > p1y) {
  2005. if (p0y > p2y) {
  2006. sign = _Math.CesiumMath.sign(p0.y);
  2007. } else {
  2008. sign = _Math.CesiumMath.sign(p2.y);
  2009. }
  2010. } else if (p1y > p2y) {
  2011. sign = _Math.CesiumMath.sign(p1.y);
  2012. } else {
  2013. sign = _Math.CesiumMath.sign(p2.y);
  2014. }
  2015. var isBehind = sign < 0.0;
  2016. offsetPointFromXZPlane(p0, isBehind);
  2017. offsetPointFromXZPlane(p1, isBehind);
  2018. offsetPointFromXZPlane(p2, isBehind);
  2019. }
  2020. var c3 = new Cartesian2.Cartesian3();
  2021. function getXZIntersectionOffsetPoints(p, p1, u1, v1) {
  2022. Cartesian2.Cartesian3.add(
  2023. p,
  2024. Cartesian2.Cartesian3.multiplyByScalar(
  2025. Cartesian2.Cartesian3.subtract(p1, p, c3),
  2026. p.y / (p.y - p1.y),
  2027. c3
  2028. ),
  2029. u1
  2030. );
  2031. Cartesian2.Cartesian3.clone(u1, v1);
  2032. offsetPointFromXZPlane(u1, true);
  2033. offsetPointFromXZPlane(v1, false);
  2034. }
  2035. var u1 = new Cartesian2.Cartesian3();
  2036. var u2 = new Cartesian2.Cartesian3();
  2037. var q1 = new Cartesian2.Cartesian3();
  2038. var q2 = new Cartesian2.Cartesian3();
  2039. var splitTriangleResult = {
  2040. positions: new Array(7),
  2041. indices: new Array(3 * 3),
  2042. };
  2043. function splitTriangle(p0, p1, p2) {
  2044. // In WGS84 coordinates, for a triangle approximately on the
  2045. // ellipsoid to cross the IDL, first it needs to be on the
  2046. // negative side of the plane x = 0.
  2047. if (p0.x >= 0.0 || p1.x >= 0.0 || p2.x >= 0.0) {
  2048. return undefined;
  2049. }
  2050. offsetTriangleFromXZPlane(p0, p1, p2);
  2051. var p0Behind = p0.y < 0.0;
  2052. var p1Behind = p1.y < 0.0;
  2053. var p2Behind = p2.y < 0.0;
  2054. var numBehind = 0;
  2055. numBehind += p0Behind ? 1 : 0;
  2056. numBehind += p1Behind ? 1 : 0;
  2057. numBehind += p2Behind ? 1 : 0;
  2058. var indices = splitTriangleResult.indices;
  2059. if (numBehind === 1) {
  2060. indices[1] = 3;
  2061. indices[2] = 4;
  2062. indices[5] = 6;
  2063. indices[7] = 6;
  2064. indices[8] = 5;
  2065. if (p0Behind) {
  2066. getXZIntersectionOffsetPoints(p0, p1, u1, q1);
  2067. getXZIntersectionOffsetPoints(p0, p2, u2, q2);
  2068. indices[0] = 0;
  2069. indices[3] = 1;
  2070. indices[4] = 2;
  2071. indices[6] = 1;
  2072. } else if (p1Behind) {
  2073. getXZIntersectionOffsetPoints(p1, p2, u1, q1);
  2074. getXZIntersectionOffsetPoints(p1, p0, u2, q2);
  2075. indices[0] = 1;
  2076. indices[3] = 2;
  2077. indices[4] = 0;
  2078. indices[6] = 2;
  2079. } else if (p2Behind) {
  2080. getXZIntersectionOffsetPoints(p2, p0, u1, q1);
  2081. getXZIntersectionOffsetPoints(p2, p1, u2, q2);
  2082. indices[0] = 2;
  2083. indices[3] = 0;
  2084. indices[4] = 1;
  2085. indices[6] = 0;
  2086. }
  2087. } else if (numBehind === 2) {
  2088. indices[2] = 4;
  2089. indices[4] = 4;
  2090. indices[5] = 3;
  2091. indices[7] = 5;
  2092. indices[8] = 6;
  2093. if (!p0Behind) {
  2094. getXZIntersectionOffsetPoints(p0, p1, u1, q1);
  2095. getXZIntersectionOffsetPoints(p0, p2, u2, q2);
  2096. indices[0] = 1;
  2097. indices[1] = 2;
  2098. indices[3] = 1;
  2099. indices[6] = 0;
  2100. } else if (!p1Behind) {
  2101. getXZIntersectionOffsetPoints(p1, p2, u1, q1);
  2102. getXZIntersectionOffsetPoints(p1, p0, u2, q2);
  2103. indices[0] = 2;
  2104. indices[1] = 0;
  2105. indices[3] = 2;
  2106. indices[6] = 1;
  2107. } else if (!p2Behind) {
  2108. getXZIntersectionOffsetPoints(p2, p0, u1, q1);
  2109. getXZIntersectionOffsetPoints(p2, p1, u2, q2);
  2110. indices[0] = 0;
  2111. indices[1] = 1;
  2112. indices[3] = 0;
  2113. indices[6] = 2;
  2114. }
  2115. }
  2116. var positions = splitTriangleResult.positions;
  2117. positions[0] = p0;
  2118. positions[1] = p1;
  2119. positions[2] = p2;
  2120. positions.length = 3;
  2121. if (numBehind === 1 || numBehind === 2) {
  2122. positions[3] = u1;
  2123. positions[4] = u2;
  2124. positions[5] = q1;
  2125. positions[6] = q2;
  2126. positions.length = 7;
  2127. }
  2128. return splitTriangleResult;
  2129. }
  2130. function updateGeometryAfterSplit(geometry, computeBoundingSphere) {
  2131. var attributes = geometry.attributes;
  2132. if (attributes.position.values.length === 0) {
  2133. return undefined;
  2134. }
  2135. for (var property in attributes) {
  2136. if (
  2137. attributes.hasOwnProperty(property) &&
  2138. when.defined(attributes[property]) &&
  2139. when.defined(attributes[property].values)
  2140. ) {
  2141. var attribute = attributes[property];
  2142. attribute.values = ComponentDatatype.ComponentDatatype.createTypedArray(
  2143. attribute.componentDatatype,
  2144. attribute.values
  2145. );
  2146. }
  2147. }
  2148. var numberOfVertices = GeometryAttribute.Geometry.computeNumberOfVertices(geometry);
  2149. geometry.indices = IndexDatatype.IndexDatatype.createTypedArray(
  2150. numberOfVertices,
  2151. geometry.indices
  2152. );
  2153. if (computeBoundingSphere) {
  2154. geometry.boundingSphere = Transforms.BoundingSphere.fromVertices(
  2155. attributes.position.values
  2156. );
  2157. }
  2158. return geometry;
  2159. }
  2160. function copyGeometryForSplit(geometry) {
  2161. var attributes = geometry.attributes;
  2162. var copiedAttributes = {};
  2163. for (var property in attributes) {
  2164. if (
  2165. attributes.hasOwnProperty(property) &&
  2166. when.defined(attributes[property]) &&
  2167. when.defined(attributes[property].values)
  2168. ) {
  2169. var attribute = attributes[property];
  2170. copiedAttributes[property] = new GeometryAttribute.GeometryAttribute({
  2171. componentDatatype: attribute.componentDatatype,
  2172. componentsPerAttribute: attribute.componentsPerAttribute,
  2173. normalize: attribute.normalize,
  2174. values: [],
  2175. });
  2176. }
  2177. }
  2178. return new GeometryAttribute.Geometry({
  2179. attributes: copiedAttributes,
  2180. indices: [],
  2181. primitiveType: geometry.primitiveType,
  2182. });
  2183. }
  2184. function updateInstanceAfterSplit(instance, westGeometry, eastGeometry) {
  2185. var computeBoundingSphere = when.defined(instance.geometry.boundingSphere);
  2186. westGeometry = updateGeometryAfterSplit(westGeometry, computeBoundingSphere);
  2187. eastGeometry = updateGeometryAfterSplit(eastGeometry, computeBoundingSphere);
  2188. if (when.defined(eastGeometry) && !when.defined(westGeometry)) {
  2189. instance.geometry = eastGeometry;
  2190. } else if (!when.defined(eastGeometry) && when.defined(westGeometry)) {
  2191. instance.geometry = westGeometry;
  2192. } else {
  2193. instance.westHemisphereGeometry = westGeometry;
  2194. instance.eastHemisphereGeometry = eastGeometry;
  2195. instance.geometry = undefined;
  2196. }
  2197. }
  2198. function generateBarycentricInterpolateFunction(
  2199. CartesianType,
  2200. numberOfComponents
  2201. ) {
  2202. var v0Scratch = new CartesianType();
  2203. var v1Scratch = new CartesianType();
  2204. var v2Scratch = new CartesianType();
  2205. return function (
  2206. i0,
  2207. i1,
  2208. i2,
  2209. coords,
  2210. sourceValues,
  2211. currentValues,
  2212. insertedIndex,
  2213. normalize
  2214. ) {
  2215. var v0 = CartesianType.fromArray(
  2216. sourceValues,
  2217. i0 * numberOfComponents,
  2218. v0Scratch
  2219. );
  2220. var v1 = CartesianType.fromArray(
  2221. sourceValues,
  2222. i1 * numberOfComponents,
  2223. v1Scratch
  2224. );
  2225. var v2 = CartesianType.fromArray(
  2226. sourceValues,
  2227. i2 * numberOfComponents,
  2228. v2Scratch
  2229. );
  2230. CartesianType.multiplyByScalar(v0, coords.x, v0);
  2231. CartesianType.multiplyByScalar(v1, coords.y, v1);
  2232. CartesianType.multiplyByScalar(v2, coords.z, v2);
  2233. var value = CartesianType.add(v0, v1, v0);
  2234. CartesianType.add(value, v2, value);
  2235. if (normalize) {
  2236. CartesianType.normalize(value, value);
  2237. }
  2238. CartesianType.pack(
  2239. value,
  2240. currentValues,
  2241. insertedIndex * numberOfComponents
  2242. );
  2243. };
  2244. }
  2245. var interpolateAndPackCartesian4 = generateBarycentricInterpolateFunction(
  2246. Transforms.Cartesian4,
  2247. 4
  2248. );
  2249. var interpolateAndPackCartesian3 = generateBarycentricInterpolateFunction(
  2250. Cartesian2.Cartesian3,
  2251. 3
  2252. );
  2253. var interpolateAndPackCartesian2 = generateBarycentricInterpolateFunction(
  2254. Cartesian2.Cartesian2,
  2255. 2
  2256. );
  2257. var interpolateAndPackBoolean = function (
  2258. i0,
  2259. i1,
  2260. i2,
  2261. coords,
  2262. sourceValues,
  2263. currentValues,
  2264. insertedIndex
  2265. ) {
  2266. var v1 = sourceValues[i0] * coords.x;
  2267. var v2 = sourceValues[i1] * coords.y;
  2268. var v3 = sourceValues[i2] * coords.z;
  2269. currentValues[insertedIndex] = v1 + v2 + v3 > _Math.CesiumMath.EPSILON6 ? 1 : 0;
  2270. };
  2271. var p0Scratch = new Cartesian2.Cartesian3();
  2272. var p1Scratch = new Cartesian2.Cartesian3();
  2273. var p2Scratch = new Cartesian2.Cartesian3();
  2274. var barycentricScratch = new Cartesian2.Cartesian3();
  2275. function computeTriangleAttributes(
  2276. i0,
  2277. i1,
  2278. i2,
  2279. point,
  2280. positions,
  2281. normals,
  2282. tangents,
  2283. bitangents,
  2284. texCoords,
  2285. extrudeDirections,
  2286. applyOffset,
  2287. currentAttributes,
  2288. customAttributeNames,
  2289. customAttributesLength,
  2290. allAttributes,
  2291. insertedIndex
  2292. ) {
  2293. if (
  2294. !when.defined(normals) &&
  2295. !when.defined(tangents) &&
  2296. !when.defined(bitangents) &&
  2297. !when.defined(texCoords) &&
  2298. !when.defined(extrudeDirections) &&
  2299. customAttributesLength === 0
  2300. ) {
  2301. return;
  2302. }
  2303. var p0 = Cartesian2.Cartesian3.fromArray(positions, i0 * 3, p0Scratch);
  2304. var p1 = Cartesian2.Cartesian3.fromArray(positions, i1 * 3, p1Scratch);
  2305. var p2 = Cartesian2.Cartesian3.fromArray(positions, i2 * 3, p2Scratch);
  2306. var coords = barycentricCoordinates(point, p0, p1, p2, barycentricScratch);
  2307. if (when.defined(normals)) {
  2308. interpolateAndPackCartesian3(
  2309. i0,
  2310. i1,
  2311. i2,
  2312. coords,
  2313. normals,
  2314. currentAttributes.normal.values,
  2315. insertedIndex,
  2316. true
  2317. );
  2318. }
  2319. if (when.defined(extrudeDirections)) {
  2320. var d0 = Cartesian2.Cartesian3.fromArray(extrudeDirections, i0 * 3, p0Scratch);
  2321. var d1 = Cartesian2.Cartesian3.fromArray(extrudeDirections, i1 * 3, p1Scratch);
  2322. var d2 = Cartesian2.Cartesian3.fromArray(extrudeDirections, i2 * 3, p2Scratch);
  2323. Cartesian2.Cartesian3.multiplyByScalar(d0, coords.x, d0);
  2324. Cartesian2.Cartesian3.multiplyByScalar(d1, coords.y, d1);
  2325. Cartesian2.Cartesian3.multiplyByScalar(d2, coords.z, d2);
  2326. var direction;
  2327. if (
  2328. !Cartesian2.Cartesian3.equals(d0, Cartesian2.Cartesian3.ZERO) ||
  2329. !Cartesian2.Cartesian3.equals(d1, Cartesian2.Cartesian3.ZERO) ||
  2330. !Cartesian2.Cartesian3.equals(d2, Cartesian2.Cartesian3.ZERO)
  2331. ) {
  2332. direction = Cartesian2.Cartesian3.add(d0, d1, d0);
  2333. Cartesian2.Cartesian3.add(direction, d2, direction);
  2334. Cartesian2.Cartesian3.normalize(direction, direction);
  2335. } else {
  2336. direction = p0Scratch;
  2337. direction.x = 0;
  2338. direction.y = 0;
  2339. direction.z = 0;
  2340. }
  2341. Cartesian2.Cartesian3.pack(
  2342. direction,
  2343. currentAttributes.extrudeDirection.values,
  2344. insertedIndex * 3
  2345. );
  2346. }
  2347. if (when.defined(applyOffset)) {
  2348. interpolateAndPackBoolean(
  2349. i0,
  2350. i1,
  2351. i2,
  2352. coords,
  2353. applyOffset,
  2354. currentAttributes.applyOffset.values,
  2355. insertedIndex
  2356. );
  2357. }
  2358. if (when.defined(tangents)) {
  2359. interpolateAndPackCartesian3(
  2360. i0,
  2361. i1,
  2362. i2,
  2363. coords,
  2364. tangents,
  2365. currentAttributes.tangent.values,
  2366. insertedIndex,
  2367. true
  2368. );
  2369. }
  2370. if (when.defined(bitangents)) {
  2371. interpolateAndPackCartesian3(
  2372. i0,
  2373. i1,
  2374. i2,
  2375. coords,
  2376. bitangents,
  2377. currentAttributes.bitangent.values,
  2378. insertedIndex,
  2379. true
  2380. );
  2381. }
  2382. if (when.defined(texCoords)) {
  2383. interpolateAndPackCartesian2(
  2384. i0,
  2385. i1,
  2386. i2,
  2387. coords,
  2388. texCoords,
  2389. currentAttributes.st.values,
  2390. insertedIndex
  2391. );
  2392. }
  2393. if (customAttributesLength > 0) {
  2394. for (var i = 0; i < customAttributesLength; i++) {
  2395. var attributeName = customAttributeNames[i];
  2396. genericInterpolate(
  2397. i0,
  2398. i1,
  2399. i2,
  2400. coords,
  2401. insertedIndex,
  2402. allAttributes[attributeName],
  2403. currentAttributes[attributeName]
  2404. );
  2405. }
  2406. }
  2407. }
  2408. function genericInterpolate(
  2409. i0,
  2410. i1,
  2411. i2,
  2412. coords,
  2413. insertedIndex,
  2414. sourceAttribute,
  2415. currentAttribute
  2416. ) {
  2417. var componentsPerAttribute = sourceAttribute.componentsPerAttribute;
  2418. var sourceValues = sourceAttribute.values;
  2419. var currentValues = currentAttribute.values;
  2420. switch (componentsPerAttribute) {
  2421. case 4:
  2422. interpolateAndPackCartesian4(
  2423. i0,
  2424. i1,
  2425. i2,
  2426. coords,
  2427. sourceValues,
  2428. currentValues,
  2429. insertedIndex,
  2430. false
  2431. );
  2432. break;
  2433. case 3:
  2434. interpolateAndPackCartesian3(
  2435. i0,
  2436. i1,
  2437. i2,
  2438. coords,
  2439. sourceValues,
  2440. currentValues,
  2441. insertedIndex,
  2442. false
  2443. );
  2444. break;
  2445. case 2:
  2446. interpolateAndPackCartesian2(
  2447. i0,
  2448. i1,
  2449. i2,
  2450. coords,
  2451. sourceValues,
  2452. currentValues,
  2453. insertedIndex,
  2454. false
  2455. );
  2456. break;
  2457. default:
  2458. currentValues[insertedIndex] =
  2459. sourceValues[i0] * coords.x +
  2460. sourceValues[i1] * coords.y +
  2461. sourceValues[i2] * coords.z;
  2462. }
  2463. }
  2464. function insertSplitPoint(
  2465. currentAttributes,
  2466. currentIndices,
  2467. currentIndexMap,
  2468. indices,
  2469. currentIndex,
  2470. point
  2471. ) {
  2472. var insertIndex = currentAttributes.position.values.length / 3;
  2473. if (currentIndex !== -1) {
  2474. var prevIndex = indices[currentIndex];
  2475. var newIndex = currentIndexMap[prevIndex];
  2476. if (newIndex === -1) {
  2477. currentIndexMap[prevIndex] = insertIndex;
  2478. currentAttributes.position.values.push(point.x, point.y, point.z);
  2479. currentIndices.push(insertIndex);
  2480. return insertIndex;
  2481. }
  2482. currentIndices.push(newIndex);
  2483. return newIndex;
  2484. }
  2485. currentAttributes.position.values.push(point.x, point.y, point.z);
  2486. currentIndices.push(insertIndex);
  2487. return insertIndex;
  2488. }
  2489. var NAMED_ATTRIBUTES = {
  2490. position: true,
  2491. normal: true,
  2492. bitangent: true,
  2493. tangent: true,
  2494. st: true,
  2495. extrudeDirection: true,
  2496. applyOffset: true,
  2497. };
  2498. function splitLongitudeTriangles(instance) {
  2499. var geometry = instance.geometry;
  2500. var attributes = geometry.attributes;
  2501. var positions = attributes.position.values;
  2502. var normals = when.defined(attributes.normal)
  2503. ? attributes.normal.values
  2504. : undefined;
  2505. var bitangents = when.defined(attributes.bitangent)
  2506. ? attributes.bitangent.values
  2507. : undefined;
  2508. var tangents = when.defined(attributes.tangent)
  2509. ? attributes.tangent.values
  2510. : undefined;
  2511. var texCoords = when.defined(attributes.st) ? attributes.st.values : undefined;
  2512. var extrudeDirections = when.defined(attributes.extrudeDirection)
  2513. ? attributes.extrudeDirection.values
  2514. : undefined;
  2515. var applyOffset = when.defined(attributes.applyOffset)
  2516. ? attributes.applyOffset.values
  2517. : undefined;
  2518. var indices = geometry.indices;
  2519. var customAttributeNames = [];
  2520. for (var attributeName in attributes) {
  2521. if (
  2522. attributes.hasOwnProperty(attributeName) &&
  2523. !NAMED_ATTRIBUTES[attributeName] &&
  2524. when.defined(attributes[attributeName])
  2525. ) {
  2526. customAttributeNames.push(attributeName);
  2527. }
  2528. }
  2529. var customAttributesLength = customAttributeNames.length;
  2530. var eastGeometry = copyGeometryForSplit(geometry);
  2531. var westGeometry = copyGeometryForSplit(geometry);
  2532. var currentAttributes;
  2533. var currentIndices;
  2534. var currentIndexMap;
  2535. var insertedIndex;
  2536. var i;
  2537. var westGeometryIndexMap = [];
  2538. westGeometryIndexMap.length = positions.length / 3;
  2539. var eastGeometryIndexMap = [];
  2540. eastGeometryIndexMap.length = positions.length / 3;
  2541. for (i = 0; i < westGeometryIndexMap.length; ++i) {
  2542. westGeometryIndexMap[i] = -1;
  2543. eastGeometryIndexMap[i] = -1;
  2544. }
  2545. var len = indices.length;
  2546. for (i = 0; i < len; i += 3) {
  2547. var i0 = indices[i];
  2548. var i1 = indices[i + 1];
  2549. var i2 = indices[i + 2];
  2550. var p0 = Cartesian2.Cartesian3.fromArray(positions, i0 * 3);
  2551. var p1 = Cartesian2.Cartesian3.fromArray(positions, i1 * 3);
  2552. var p2 = Cartesian2.Cartesian3.fromArray(positions, i2 * 3);
  2553. var result = splitTriangle(p0, p1, p2);
  2554. if (when.defined(result) && result.positions.length > 3) {
  2555. var resultPositions = result.positions;
  2556. var resultIndices = result.indices;
  2557. var resultLength = resultIndices.length;
  2558. for (var j = 0; j < resultLength; ++j) {
  2559. var resultIndex = resultIndices[j];
  2560. var point = resultPositions[resultIndex];
  2561. if (point.y < 0.0) {
  2562. currentAttributes = westGeometry.attributes;
  2563. currentIndices = westGeometry.indices;
  2564. currentIndexMap = westGeometryIndexMap;
  2565. } else {
  2566. currentAttributes = eastGeometry.attributes;
  2567. currentIndices = eastGeometry.indices;
  2568. currentIndexMap = eastGeometryIndexMap;
  2569. }
  2570. insertedIndex = insertSplitPoint(
  2571. currentAttributes,
  2572. currentIndices,
  2573. currentIndexMap,
  2574. indices,
  2575. resultIndex < 3 ? i + resultIndex : -1,
  2576. point
  2577. );
  2578. computeTriangleAttributes(
  2579. i0,
  2580. i1,
  2581. i2,
  2582. point,
  2583. positions,
  2584. normals,
  2585. tangents,
  2586. bitangents,
  2587. texCoords,
  2588. extrudeDirections,
  2589. applyOffset,
  2590. currentAttributes,
  2591. customAttributeNames,
  2592. customAttributesLength,
  2593. attributes,
  2594. insertedIndex
  2595. );
  2596. }
  2597. } else {
  2598. if (when.defined(result)) {
  2599. p0 = result.positions[0];
  2600. p1 = result.positions[1];
  2601. p2 = result.positions[2];
  2602. }
  2603. if (p0.y < 0.0) {
  2604. currentAttributes = westGeometry.attributes;
  2605. currentIndices = westGeometry.indices;
  2606. currentIndexMap = westGeometryIndexMap;
  2607. } else {
  2608. currentAttributes = eastGeometry.attributes;
  2609. currentIndices = eastGeometry.indices;
  2610. currentIndexMap = eastGeometryIndexMap;
  2611. }
  2612. insertedIndex = insertSplitPoint(
  2613. currentAttributes,
  2614. currentIndices,
  2615. currentIndexMap,
  2616. indices,
  2617. i,
  2618. p0
  2619. );
  2620. computeTriangleAttributes(
  2621. i0,
  2622. i1,
  2623. i2,
  2624. p0,
  2625. positions,
  2626. normals,
  2627. tangents,
  2628. bitangents,
  2629. texCoords,
  2630. extrudeDirections,
  2631. applyOffset,
  2632. currentAttributes,
  2633. customAttributeNames,
  2634. customAttributesLength,
  2635. attributes,
  2636. insertedIndex
  2637. );
  2638. insertedIndex = insertSplitPoint(
  2639. currentAttributes,
  2640. currentIndices,
  2641. currentIndexMap,
  2642. indices,
  2643. i + 1,
  2644. p1
  2645. );
  2646. computeTriangleAttributes(
  2647. i0,
  2648. i1,
  2649. i2,
  2650. p1,
  2651. positions,
  2652. normals,
  2653. tangents,
  2654. bitangents,
  2655. texCoords,
  2656. extrudeDirections,
  2657. applyOffset,
  2658. currentAttributes,
  2659. customAttributeNames,
  2660. customAttributesLength,
  2661. attributes,
  2662. insertedIndex
  2663. );
  2664. insertedIndex = insertSplitPoint(
  2665. currentAttributes,
  2666. currentIndices,
  2667. currentIndexMap,
  2668. indices,
  2669. i + 2,
  2670. p2
  2671. );
  2672. computeTriangleAttributes(
  2673. i0,
  2674. i1,
  2675. i2,
  2676. p2,
  2677. positions,
  2678. normals,
  2679. tangents,
  2680. bitangents,
  2681. texCoords,
  2682. extrudeDirections,
  2683. applyOffset,
  2684. currentAttributes,
  2685. customAttributeNames,
  2686. customAttributesLength,
  2687. attributes,
  2688. insertedIndex
  2689. );
  2690. }
  2691. }
  2692. updateInstanceAfterSplit(instance, westGeometry, eastGeometry);
  2693. }
  2694. var xzPlane = Plane.Plane.fromPointNormal(Cartesian2.Cartesian3.ZERO, Cartesian2.Cartesian3.UNIT_Y);
  2695. var offsetScratch = new Cartesian2.Cartesian3();
  2696. var offsetPointScratch = new Cartesian2.Cartesian3();
  2697. function computeLineAttributes(
  2698. i0,
  2699. i1,
  2700. point,
  2701. positions,
  2702. insertIndex,
  2703. currentAttributes,
  2704. applyOffset
  2705. ) {
  2706. if (!when.defined(applyOffset)) {
  2707. return;
  2708. }
  2709. var p0 = Cartesian2.Cartesian3.fromArray(positions, i0 * 3, p0Scratch);
  2710. if (Cartesian2.Cartesian3.equalsEpsilon(p0, point, _Math.CesiumMath.EPSILON10)) {
  2711. currentAttributes.applyOffset.values[insertIndex] = applyOffset[i0];
  2712. } else {
  2713. currentAttributes.applyOffset.values[insertIndex] = applyOffset[i1];
  2714. }
  2715. }
  2716. function splitLongitudeLines(instance) {
  2717. var geometry = instance.geometry;
  2718. var attributes = geometry.attributes;
  2719. var positions = attributes.position.values;
  2720. var applyOffset = when.defined(attributes.applyOffset)
  2721. ? attributes.applyOffset.values
  2722. : undefined;
  2723. var indices = geometry.indices;
  2724. var eastGeometry = copyGeometryForSplit(geometry);
  2725. var westGeometry = copyGeometryForSplit(geometry);
  2726. var i;
  2727. var length = indices.length;
  2728. var westGeometryIndexMap = [];
  2729. westGeometryIndexMap.length = positions.length / 3;
  2730. var eastGeometryIndexMap = [];
  2731. eastGeometryIndexMap.length = positions.length / 3;
  2732. for (i = 0; i < westGeometryIndexMap.length; ++i) {
  2733. westGeometryIndexMap[i] = -1;
  2734. eastGeometryIndexMap[i] = -1;
  2735. }
  2736. for (i = 0; i < length; i += 2) {
  2737. var i0 = indices[i];
  2738. var i1 = indices[i + 1];
  2739. var p0 = Cartesian2.Cartesian3.fromArray(positions, i0 * 3, p0Scratch);
  2740. var p1 = Cartesian2.Cartesian3.fromArray(positions, i1 * 3, p1Scratch);
  2741. var insertIndex;
  2742. if (Math.abs(p0.y) < _Math.CesiumMath.EPSILON6) {
  2743. if (p0.y < 0.0) {
  2744. p0.y = -_Math.CesiumMath.EPSILON6;
  2745. } else {
  2746. p0.y = _Math.CesiumMath.EPSILON6;
  2747. }
  2748. }
  2749. if (Math.abs(p1.y) < _Math.CesiumMath.EPSILON6) {
  2750. if (p1.y < 0.0) {
  2751. p1.y = -_Math.CesiumMath.EPSILON6;
  2752. } else {
  2753. p1.y = _Math.CesiumMath.EPSILON6;
  2754. }
  2755. }
  2756. var p0Attributes = eastGeometry.attributes;
  2757. var p0Indices = eastGeometry.indices;
  2758. var p0IndexMap = eastGeometryIndexMap;
  2759. var p1Attributes = westGeometry.attributes;
  2760. var p1Indices = westGeometry.indices;
  2761. var p1IndexMap = westGeometryIndexMap;
  2762. var intersection = IntersectionTests.IntersectionTests.lineSegmentPlane(
  2763. p0,
  2764. p1,
  2765. xzPlane,
  2766. p2Scratch
  2767. );
  2768. if (when.defined(intersection)) {
  2769. // move point on the xz-plane slightly away from the plane
  2770. var offset = Cartesian2.Cartesian3.multiplyByScalar(
  2771. Cartesian2.Cartesian3.UNIT_Y,
  2772. 5.0 * _Math.CesiumMath.EPSILON9,
  2773. offsetScratch
  2774. );
  2775. if (p0.y < 0.0) {
  2776. Cartesian2.Cartesian3.negate(offset, offset);
  2777. p0Attributes = westGeometry.attributes;
  2778. p0Indices = westGeometry.indices;
  2779. p0IndexMap = westGeometryIndexMap;
  2780. p1Attributes = eastGeometry.attributes;
  2781. p1Indices = eastGeometry.indices;
  2782. p1IndexMap = eastGeometryIndexMap;
  2783. }
  2784. var offsetPoint = Cartesian2.Cartesian3.add(
  2785. intersection,
  2786. offset,
  2787. offsetPointScratch
  2788. );
  2789. insertIndex = insertSplitPoint(
  2790. p0Attributes,
  2791. p0Indices,
  2792. p0IndexMap,
  2793. indices,
  2794. i,
  2795. p0
  2796. );
  2797. computeLineAttributes(
  2798. i0,
  2799. i1,
  2800. p0,
  2801. positions,
  2802. insertIndex,
  2803. p0Attributes,
  2804. applyOffset
  2805. );
  2806. insertIndex = insertSplitPoint(
  2807. p0Attributes,
  2808. p0Indices,
  2809. p0IndexMap,
  2810. indices,
  2811. -1,
  2812. offsetPoint
  2813. );
  2814. computeLineAttributes(
  2815. i0,
  2816. i1,
  2817. offsetPoint,
  2818. positions,
  2819. insertIndex,
  2820. p0Attributes,
  2821. applyOffset
  2822. );
  2823. Cartesian2.Cartesian3.negate(offset, offset);
  2824. Cartesian2.Cartesian3.add(intersection, offset, offsetPoint);
  2825. insertIndex = insertSplitPoint(
  2826. p1Attributes,
  2827. p1Indices,
  2828. p1IndexMap,
  2829. indices,
  2830. -1,
  2831. offsetPoint
  2832. );
  2833. computeLineAttributes(
  2834. i0,
  2835. i1,
  2836. offsetPoint,
  2837. positions,
  2838. insertIndex,
  2839. p1Attributes,
  2840. applyOffset
  2841. );
  2842. insertIndex = insertSplitPoint(
  2843. p1Attributes,
  2844. p1Indices,
  2845. p1IndexMap,
  2846. indices,
  2847. i + 1,
  2848. p1
  2849. );
  2850. computeLineAttributes(
  2851. i0,
  2852. i1,
  2853. p1,
  2854. positions,
  2855. insertIndex,
  2856. p1Attributes,
  2857. applyOffset
  2858. );
  2859. } else {
  2860. var currentAttributes;
  2861. var currentIndices;
  2862. var currentIndexMap;
  2863. if (p0.y < 0.0) {
  2864. currentAttributes = westGeometry.attributes;
  2865. currentIndices = westGeometry.indices;
  2866. currentIndexMap = westGeometryIndexMap;
  2867. } else {
  2868. currentAttributes = eastGeometry.attributes;
  2869. currentIndices = eastGeometry.indices;
  2870. currentIndexMap = eastGeometryIndexMap;
  2871. }
  2872. insertIndex = insertSplitPoint(
  2873. currentAttributes,
  2874. currentIndices,
  2875. currentIndexMap,
  2876. indices,
  2877. i,
  2878. p0
  2879. );
  2880. computeLineAttributes(
  2881. i0,
  2882. i1,
  2883. p0,
  2884. positions,
  2885. insertIndex,
  2886. currentAttributes,
  2887. applyOffset
  2888. );
  2889. insertIndex = insertSplitPoint(
  2890. currentAttributes,
  2891. currentIndices,
  2892. currentIndexMap,
  2893. indices,
  2894. i + 1,
  2895. p1
  2896. );
  2897. computeLineAttributes(
  2898. i0,
  2899. i1,
  2900. p1,
  2901. positions,
  2902. insertIndex,
  2903. currentAttributes,
  2904. applyOffset
  2905. );
  2906. }
  2907. }
  2908. updateInstanceAfterSplit(instance, westGeometry, eastGeometry);
  2909. }
  2910. var cartesian2Scratch0 = new Cartesian2.Cartesian2();
  2911. var cartesian2Scratch1 = new Cartesian2.Cartesian2();
  2912. var cartesian3Scratch0 = new Cartesian2.Cartesian3();
  2913. var cartesian3Scratch2 = new Cartesian2.Cartesian3();
  2914. var cartesian3Scratch3 = new Cartesian2.Cartesian3();
  2915. var cartesian3Scratch4 = new Cartesian2.Cartesian3();
  2916. var cartesian3Scratch5 = new Cartesian2.Cartesian3();
  2917. var cartesian3Scratch6 = new Cartesian2.Cartesian3();
  2918. var cartesian4Scratch0 = new Transforms.Cartesian4();
  2919. function updateAdjacencyAfterSplit(geometry) {
  2920. var attributes = geometry.attributes;
  2921. var positions = attributes.position.values;
  2922. var prevPositions = attributes.prevPosition.values;
  2923. var nextPositions = attributes.nextPosition.values;
  2924. var length = positions.length;
  2925. for (var j = 0; j < length; j += 3) {
  2926. var position = Cartesian2.Cartesian3.unpack(positions, j, cartesian3Scratch0);
  2927. if (position.x > 0.0) {
  2928. continue;
  2929. }
  2930. var prevPosition = Cartesian2.Cartesian3.unpack(prevPositions, j, cartesian3Scratch2);
  2931. if (
  2932. (position.y < 0.0 && prevPosition.y > 0.0) ||
  2933. (position.y > 0.0 && prevPosition.y < 0.0)
  2934. ) {
  2935. if (j - 3 > 0) {
  2936. prevPositions[j] = positions[j - 3];
  2937. prevPositions[j + 1] = positions[j - 2];
  2938. prevPositions[j + 2] = positions[j - 1];
  2939. } else {
  2940. Cartesian2.Cartesian3.pack(position, prevPositions, j);
  2941. }
  2942. }
  2943. var nextPosition = Cartesian2.Cartesian3.unpack(nextPositions, j, cartesian3Scratch3);
  2944. if (
  2945. (position.y < 0.0 && nextPosition.y > 0.0) ||
  2946. (position.y > 0.0 && nextPosition.y < 0.0)
  2947. ) {
  2948. if (j + 3 < length) {
  2949. nextPositions[j] = positions[j + 3];
  2950. nextPositions[j + 1] = positions[j + 4];
  2951. nextPositions[j + 2] = positions[j + 5];
  2952. } else {
  2953. Cartesian2.Cartesian3.pack(position, nextPositions, j);
  2954. }
  2955. }
  2956. }
  2957. }
  2958. var offsetScalar = 5.0 * _Math.CesiumMath.EPSILON9;
  2959. var coplanarOffset = _Math.CesiumMath.EPSILON6;
  2960. function splitLongitudePolyline(instance) {
  2961. var geometry = instance.geometry;
  2962. var attributes = geometry.attributes;
  2963. var positions = attributes.position.values;
  2964. var prevPositions = attributes.prevPosition.values;
  2965. var nextPositions = attributes.nextPosition.values;
  2966. var expandAndWidths = attributes.expandAndWidth.values;
  2967. var texCoords = when.defined(attributes.st) ? attributes.st.values : undefined;
  2968. var colors = when.defined(attributes.color) ? attributes.color.values : undefined;
  2969. var eastGeometry = copyGeometryForSplit(geometry);
  2970. var westGeometry = copyGeometryForSplit(geometry);
  2971. var i;
  2972. var j;
  2973. var index;
  2974. var intersectionFound = false;
  2975. var length = positions.length / 3;
  2976. for (i = 0; i < length; i += 4) {
  2977. var i0 = i;
  2978. var i2 = i + 2;
  2979. var p0 = Cartesian2.Cartesian3.fromArray(positions, i0 * 3, cartesian3Scratch0);
  2980. var p2 = Cartesian2.Cartesian3.fromArray(positions, i2 * 3, cartesian3Scratch2);
  2981. // Offset points that are close to the 180 longitude and change the previous/next point
  2982. // to be the same offset point so it can be projected to 2D. There is special handling in the
  2983. // shader for when position == prevPosition || position == nextPosition.
  2984. if (Math.abs(p0.y) < coplanarOffset) {
  2985. p0.y = coplanarOffset * (p2.y < 0.0 ? -1.0 : 1.0);
  2986. positions[i * 3 + 1] = p0.y;
  2987. positions[(i + 1) * 3 + 1] = p0.y;
  2988. for (j = i0 * 3; j < i0 * 3 + 4 * 3; j += 3) {
  2989. prevPositions[j] = positions[i * 3];
  2990. prevPositions[j + 1] = positions[i * 3 + 1];
  2991. prevPositions[j + 2] = positions[i * 3 + 2];
  2992. }
  2993. }
  2994. // Do the same but for when the line crosses 180 longitude in the opposite direction.
  2995. if (Math.abs(p2.y) < coplanarOffset) {
  2996. p2.y = coplanarOffset * (p0.y < 0.0 ? -1.0 : 1.0);
  2997. positions[(i + 2) * 3 + 1] = p2.y;
  2998. positions[(i + 3) * 3 + 1] = p2.y;
  2999. for (j = i0 * 3; j < i0 * 3 + 4 * 3; j += 3) {
  3000. nextPositions[j] = positions[(i + 2) * 3];
  3001. nextPositions[j + 1] = positions[(i + 2) * 3 + 1];
  3002. nextPositions[j + 2] = positions[(i + 2) * 3 + 2];
  3003. }
  3004. }
  3005. var p0Attributes = eastGeometry.attributes;
  3006. var p0Indices = eastGeometry.indices;
  3007. var p2Attributes = westGeometry.attributes;
  3008. var p2Indices = westGeometry.indices;
  3009. var intersection = IntersectionTests.IntersectionTests.lineSegmentPlane(
  3010. p0,
  3011. p2,
  3012. xzPlane,
  3013. cartesian3Scratch4
  3014. );
  3015. if (when.defined(intersection)) {
  3016. intersectionFound = true;
  3017. // move point on the xz-plane slightly away from the plane
  3018. var offset = Cartesian2.Cartesian3.multiplyByScalar(
  3019. Cartesian2.Cartesian3.UNIT_Y,
  3020. offsetScalar,
  3021. cartesian3Scratch5
  3022. );
  3023. if (p0.y < 0.0) {
  3024. Cartesian2.Cartesian3.negate(offset, offset);
  3025. p0Attributes = westGeometry.attributes;
  3026. p0Indices = westGeometry.indices;
  3027. p2Attributes = eastGeometry.attributes;
  3028. p2Indices = eastGeometry.indices;
  3029. }
  3030. var offsetPoint = Cartesian2.Cartesian3.add(
  3031. intersection,
  3032. offset,
  3033. cartesian3Scratch6
  3034. );
  3035. p0Attributes.position.values.push(p0.x, p0.y, p0.z, p0.x, p0.y, p0.z);
  3036. p0Attributes.position.values.push(
  3037. offsetPoint.x,
  3038. offsetPoint.y,
  3039. offsetPoint.z
  3040. );
  3041. p0Attributes.position.values.push(
  3042. offsetPoint.x,
  3043. offsetPoint.y,
  3044. offsetPoint.z
  3045. );
  3046. p0Attributes.prevPosition.values.push(
  3047. prevPositions[i0 * 3],
  3048. prevPositions[i0 * 3 + 1],
  3049. prevPositions[i0 * 3 + 2]
  3050. );
  3051. p0Attributes.prevPosition.values.push(
  3052. prevPositions[i0 * 3 + 3],
  3053. prevPositions[i0 * 3 + 4],
  3054. prevPositions[i0 * 3 + 5]
  3055. );
  3056. p0Attributes.prevPosition.values.push(p0.x, p0.y, p0.z, p0.x, p0.y, p0.z);
  3057. p0Attributes.nextPosition.values.push(
  3058. offsetPoint.x,
  3059. offsetPoint.y,
  3060. offsetPoint.z
  3061. );
  3062. p0Attributes.nextPosition.values.push(
  3063. offsetPoint.x,
  3064. offsetPoint.y,
  3065. offsetPoint.z
  3066. );
  3067. p0Attributes.nextPosition.values.push(
  3068. offsetPoint.x,
  3069. offsetPoint.y,
  3070. offsetPoint.z
  3071. );
  3072. p0Attributes.nextPosition.values.push(
  3073. offsetPoint.x,
  3074. offsetPoint.y,
  3075. offsetPoint.z
  3076. );
  3077. Cartesian2.Cartesian3.negate(offset, offset);
  3078. Cartesian2.Cartesian3.add(intersection, offset, offsetPoint);
  3079. p2Attributes.position.values.push(
  3080. offsetPoint.x,
  3081. offsetPoint.y,
  3082. offsetPoint.z
  3083. );
  3084. p2Attributes.position.values.push(
  3085. offsetPoint.x,
  3086. offsetPoint.y,
  3087. offsetPoint.z
  3088. );
  3089. p2Attributes.position.values.push(p2.x, p2.y, p2.z, p2.x, p2.y, p2.z);
  3090. p2Attributes.prevPosition.values.push(
  3091. offsetPoint.x,
  3092. offsetPoint.y,
  3093. offsetPoint.z
  3094. );
  3095. p2Attributes.prevPosition.values.push(
  3096. offsetPoint.x,
  3097. offsetPoint.y,
  3098. offsetPoint.z
  3099. );
  3100. p2Attributes.prevPosition.values.push(
  3101. offsetPoint.x,
  3102. offsetPoint.y,
  3103. offsetPoint.z
  3104. );
  3105. p2Attributes.prevPosition.values.push(
  3106. offsetPoint.x,
  3107. offsetPoint.y,
  3108. offsetPoint.z
  3109. );
  3110. p2Attributes.nextPosition.values.push(p2.x, p2.y, p2.z, p2.x, p2.y, p2.z);
  3111. p2Attributes.nextPosition.values.push(
  3112. nextPositions[i2 * 3],
  3113. nextPositions[i2 * 3 + 1],
  3114. nextPositions[i2 * 3 + 2]
  3115. );
  3116. p2Attributes.nextPosition.values.push(
  3117. nextPositions[i2 * 3 + 3],
  3118. nextPositions[i2 * 3 + 4],
  3119. nextPositions[i2 * 3 + 5]
  3120. );
  3121. var ew0 = Cartesian2.Cartesian2.fromArray(
  3122. expandAndWidths,
  3123. i0 * 2,
  3124. cartesian2Scratch0
  3125. );
  3126. var width = Math.abs(ew0.y);
  3127. p0Attributes.expandAndWidth.values.push(-1, width, 1, width);
  3128. p0Attributes.expandAndWidth.values.push(-1, -width, 1, -width);
  3129. p2Attributes.expandAndWidth.values.push(-1, width, 1, width);
  3130. p2Attributes.expandAndWidth.values.push(-1, -width, 1, -width);
  3131. var t = Cartesian2.Cartesian3.magnitudeSquared(
  3132. Cartesian2.Cartesian3.subtract(intersection, p0, cartesian3Scratch3)
  3133. );
  3134. t /= Cartesian2.Cartesian3.magnitudeSquared(
  3135. Cartesian2.Cartesian3.subtract(p2, p0, cartesian3Scratch3)
  3136. );
  3137. if (when.defined(colors)) {
  3138. var c0 = Transforms.Cartesian4.fromArray(colors, i0 * 4, cartesian4Scratch0);
  3139. var c2 = Transforms.Cartesian4.fromArray(colors, i2 * 4, cartesian4Scratch0);
  3140. var r = _Math.CesiumMath.lerp(c0.x, c2.x, t);
  3141. var g = _Math.CesiumMath.lerp(c0.y, c2.y, t);
  3142. var b = _Math.CesiumMath.lerp(c0.z, c2.z, t);
  3143. var a = _Math.CesiumMath.lerp(c0.w, c2.w, t);
  3144. for (j = i0 * 4; j < i0 * 4 + 2 * 4; ++j) {
  3145. p0Attributes.color.values.push(colors[j]);
  3146. }
  3147. p0Attributes.color.values.push(r, g, b, a);
  3148. p0Attributes.color.values.push(r, g, b, a);
  3149. p2Attributes.color.values.push(r, g, b, a);
  3150. p2Attributes.color.values.push(r, g, b, a);
  3151. for (j = i2 * 4; j < i2 * 4 + 2 * 4; ++j) {
  3152. p2Attributes.color.values.push(colors[j]);
  3153. }
  3154. }
  3155. if (when.defined(texCoords)) {
  3156. var s0 = Cartesian2.Cartesian2.fromArray(texCoords, i0 * 2, cartesian2Scratch0);
  3157. var s3 = Cartesian2.Cartesian2.fromArray(
  3158. texCoords,
  3159. (i + 3) * 2,
  3160. cartesian2Scratch1
  3161. );
  3162. var sx = _Math.CesiumMath.lerp(s0.x, s3.x, t);
  3163. for (j = i0 * 2; j < i0 * 2 + 2 * 2; ++j) {
  3164. p0Attributes.st.values.push(texCoords[j]);
  3165. }
  3166. p0Attributes.st.values.push(sx, s0.y);
  3167. p0Attributes.st.values.push(sx, s3.y);
  3168. p2Attributes.st.values.push(sx, s0.y);
  3169. p2Attributes.st.values.push(sx, s3.y);
  3170. for (j = i2 * 2; j < i2 * 2 + 2 * 2; ++j) {
  3171. p2Attributes.st.values.push(texCoords[j]);
  3172. }
  3173. }
  3174. index = p0Attributes.position.values.length / 3 - 4;
  3175. p0Indices.push(index, index + 2, index + 1);
  3176. p0Indices.push(index + 1, index + 2, index + 3);
  3177. index = p2Attributes.position.values.length / 3 - 4;
  3178. p2Indices.push(index, index + 2, index + 1);
  3179. p2Indices.push(index + 1, index + 2, index + 3);
  3180. } else {
  3181. var currentAttributes;
  3182. var currentIndices;
  3183. if (p0.y < 0.0) {
  3184. currentAttributes = westGeometry.attributes;
  3185. currentIndices = westGeometry.indices;
  3186. } else {
  3187. currentAttributes = eastGeometry.attributes;
  3188. currentIndices = eastGeometry.indices;
  3189. }
  3190. currentAttributes.position.values.push(p0.x, p0.y, p0.z);
  3191. currentAttributes.position.values.push(p0.x, p0.y, p0.z);
  3192. currentAttributes.position.values.push(p2.x, p2.y, p2.z);
  3193. currentAttributes.position.values.push(p2.x, p2.y, p2.z);
  3194. for (j = i * 3; j < i * 3 + 4 * 3; ++j) {
  3195. currentAttributes.prevPosition.values.push(prevPositions[j]);
  3196. currentAttributes.nextPosition.values.push(nextPositions[j]);
  3197. }
  3198. for (j = i * 2; j < i * 2 + 4 * 2; ++j) {
  3199. currentAttributes.expandAndWidth.values.push(expandAndWidths[j]);
  3200. if (when.defined(texCoords)) {
  3201. currentAttributes.st.values.push(texCoords[j]);
  3202. }
  3203. }
  3204. if (when.defined(colors)) {
  3205. for (j = i * 4; j < i * 4 + 4 * 4; ++j) {
  3206. currentAttributes.color.values.push(colors[j]);
  3207. }
  3208. }
  3209. index = currentAttributes.position.values.length / 3 - 4;
  3210. currentIndices.push(index, index + 2, index + 1);
  3211. currentIndices.push(index + 1, index + 2, index + 3);
  3212. }
  3213. }
  3214. if (intersectionFound) {
  3215. updateAdjacencyAfterSplit(westGeometry);
  3216. updateAdjacencyAfterSplit(eastGeometry);
  3217. }
  3218. updateInstanceAfterSplit(instance, westGeometry, eastGeometry);
  3219. }
  3220. /**
  3221. * Splits the instances's geometry, by introducing new vertices and indices,that
  3222. * intersect the International Date Line and Prime Meridian so that no primitives cross longitude
  3223. * -180/180 degrees. This is not required for 3D drawing, but is required for
  3224. * correcting drawing in 2D and Columbus view.
  3225. *
  3226. * @private
  3227. *
  3228. * @param {GeometryInstance} instance The instance to modify.
  3229. * @returns {GeometryInstance} The modified <code>instance</code> argument, with it's geometry split at the International Date Line.
  3230. *
  3231. * @example
  3232. * instance = Cesium.GeometryPipeline.splitLongitude(instance);
  3233. */
  3234. GeometryPipeline.splitLongitude = function (instance) {
  3235. //>>includeStart('debug', pragmas.debug);
  3236. if (!when.defined(instance)) {
  3237. throw new Check.DeveloperError("instance is required.");
  3238. }
  3239. //>>includeEnd('debug');
  3240. var geometry = instance.geometry;
  3241. var boundingSphere = geometry.boundingSphere;
  3242. if (when.defined(boundingSphere)) {
  3243. var minX = boundingSphere.center.x - boundingSphere.radius;
  3244. if (
  3245. minX > 0 ||
  3246. Transforms.BoundingSphere.intersectPlane(boundingSphere, Plane.Plane.ORIGIN_ZX_PLANE) !==
  3247. Transforms.Intersect.INTERSECTING
  3248. ) {
  3249. return instance;
  3250. }
  3251. }
  3252. if (geometry.geometryType !== GeometryAttribute.GeometryType.NONE) {
  3253. switch (geometry.geometryType) {
  3254. case GeometryAttribute.GeometryType.POLYLINES:
  3255. splitLongitudePolyline(instance);
  3256. break;
  3257. case GeometryAttribute.GeometryType.TRIANGLES:
  3258. splitLongitudeTriangles(instance);
  3259. break;
  3260. case GeometryAttribute.GeometryType.LINES:
  3261. splitLongitudeLines(instance);
  3262. break;
  3263. }
  3264. } else {
  3265. indexPrimitive(geometry);
  3266. if (geometry.primitiveType === GeometryAttribute.PrimitiveType.TRIANGLES) {
  3267. splitLongitudeTriangles(instance);
  3268. } else if (geometry.primitiveType === GeometryAttribute.PrimitiveType.LINES) {
  3269. splitLongitudeLines(instance);
  3270. }
  3271. }
  3272. return instance;
  3273. };
  3274. exports.GeometryPipeline = GeometryPipeline;
  3275. });
  3276. //# sourceMappingURL=GeometryPipeline-466ad516.js.map