Geometry.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  1. /**
  2. * @author mr.doob / http://mrdoob.com/
  3. * @author kile / http://kile.stravaganza.org/
  4. * @author alteredq / http://alteredqualia.com/
  5. * @author mikael emtinger / http://gomo.se/
  6. */
  7. THREE.Geometry = function () {
  8. this.id = "Geometry" + THREE.GeometryIdCounter ++;
  9. this.vertices = [];
  10. this.colors = []; // one-to-one vertex colors, used in ParticleSystem, Line and Ribbon
  11. this.faces = [];
  12. this.edges = [];
  13. this.faceUvs = [[]];
  14. this.faceVertexUvs = [[]];
  15. this.morphTargets = [];
  16. this.morphColors = [];
  17. this.skinWeights = [];
  18. this.skinIndices = [];
  19. this.boundingBox = null;
  20. this.boundingSphere = null;
  21. this.hasTangents = false;
  22. this.dynamic = false; // unless set to true the *Arrays will be deleted once sent to a buffer.
  23. };
  24. THREE.Geometry.prototype = {
  25. constructor : THREE.Geometry,
  26. computeCentroids: function () {
  27. var f, fl, face;
  28. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  29. face = this.faces[ f ];
  30. face.centroid.set( 0, 0, 0 );
  31. if ( face instanceof THREE.Face3 ) {
  32. face.centroid.addSelf( this.vertices[ face.a ].position );
  33. face.centroid.addSelf( this.vertices[ face.b ].position );
  34. face.centroid.addSelf( this.vertices[ face.c ].position );
  35. face.centroid.divideScalar( 3 );
  36. } else if ( face instanceof THREE.Face4 ) {
  37. face.centroid.addSelf( this.vertices[ face.a ].position );
  38. face.centroid.addSelf( this.vertices[ face.b ].position );
  39. face.centroid.addSelf( this.vertices[ face.c ].position );
  40. face.centroid.addSelf( this.vertices[ face.d ].position );
  41. face.centroid.divideScalar( 4 );
  42. }
  43. }
  44. },
  45. computeFaceNormals: function ( useVertexNormals ) {
  46. var n, nl, v, vl, vertex, f, fl, face, vA, vB, vC,
  47. cb = new THREE.Vector3(), ab = new THREE.Vector3();
  48. /*
  49. for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {
  50. vertex = this.vertices[ v ];
  51. vertex.normal.set( 0, 0, 0 );
  52. }
  53. */
  54. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  55. face = this.faces[ f ];
  56. if ( useVertexNormals && face.vertexNormals.length ) {
  57. cb.set( 0, 0, 0 );
  58. for ( n = 0, nl = face.vertexNormals.length; n < nl; n++ ) {
  59. cb.addSelf( face.vertexNormals[n] );
  60. }
  61. cb.divideScalar( 3 );
  62. if ( ! cb.isZero() ) {
  63. cb.normalize();
  64. }
  65. face.normal.copy( cb );
  66. } else {
  67. vA = this.vertices[ face.a ];
  68. vB = this.vertices[ face.b ];
  69. vC = this.vertices[ face.c ];
  70. cb.sub( vC.position, vB.position );
  71. ab.sub( vA.position, vB.position );
  72. cb.crossSelf( ab );
  73. if ( !cb.isZero() ) {
  74. cb.normalize();
  75. }
  76. face.normal.copy( cb );
  77. }
  78. }
  79. },
  80. computeVertexNormals: function () {
  81. var v, vl, f, fl, face, vertices;
  82. // create internal buffers for reuse when calling this method repeatedly
  83. // (otherwise memory allocation / deallocation every frame is big resource hog)
  84. if ( this.__tmpVertices == undefined ) {
  85. this.__tmpVertices = new Array( this.vertices.length );
  86. vertices = this.__tmpVertices;
  87. for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {
  88. vertices[ v ] = new THREE.Vector3();
  89. }
  90. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  91. face = this.faces[ f ];
  92. if ( face instanceof THREE.Face3 ) {
  93. face.vertexNormals = [ new THREE.Vector3(), new THREE.Vector3(), new THREE.Vector3() ];
  94. } else if ( face instanceof THREE.Face4 ) {
  95. face.vertexNormals = [ new THREE.Vector3(), new THREE.Vector3(), new THREE.Vector3(), new THREE.Vector3() ];
  96. }
  97. }
  98. } else {
  99. vertices = this.__tmpVertices;
  100. for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {
  101. vertices[ v ].set( 0, 0, 0 );
  102. }
  103. }
  104. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  105. face = this.faces[ f ];
  106. if ( face instanceof THREE.Face3 ) {
  107. vertices[ face.a ].addSelf( face.normal );
  108. vertices[ face.b ].addSelf( face.normal );
  109. vertices[ face.c ].addSelf( face.normal );
  110. } else if ( face instanceof THREE.Face4 ) {
  111. vertices[ face.a ].addSelf( face.normal );
  112. vertices[ face.b ].addSelf( face.normal );
  113. vertices[ face.c ].addSelf( face.normal );
  114. vertices[ face.d ].addSelf( face.normal );
  115. }
  116. }
  117. for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {
  118. vertices[ v ].normalize();
  119. }
  120. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  121. face = this.faces[ f ];
  122. if ( face instanceof THREE.Face3 ) {
  123. face.vertexNormals[ 0 ].copy( vertices[ face.a ] );
  124. face.vertexNormals[ 1 ].copy( vertices[ face.b ] );
  125. face.vertexNormals[ 2 ].copy( vertices[ face.c ] );
  126. } else if ( face instanceof THREE.Face4 ) {
  127. face.vertexNormals[ 0 ].copy( vertices[ face.a ] );
  128. face.vertexNormals[ 1 ].copy( vertices[ face.b ] );
  129. face.vertexNormals[ 2 ].copy( vertices[ face.c ] );
  130. face.vertexNormals[ 3 ].copy( vertices[ face.d ] );
  131. }
  132. }
  133. },
  134. computeTangents: function () {
  135. // based on http://www.terathon.com/code/tangent.html
  136. // tangents go to vertices
  137. var f, fl, v, vl, i, il, vertexIndex,
  138. face, uv, vA, vB, vC, uvA, uvB, uvC,
  139. x1, x2, y1, y2, z1, z2,
  140. s1, s2, t1, t2, r, t, test,
  141. tan1 = [], tan2 = [],
  142. sdir = new THREE.Vector3(), tdir = new THREE.Vector3(),
  143. tmp = new THREE.Vector3(), tmp2 = new THREE.Vector3(),
  144. n = new THREE.Vector3(), w;
  145. for ( v = 0, vl = this.vertices.length; v < vl; v ++ ) {
  146. tan1[ v ] = new THREE.Vector3();
  147. tan2[ v ] = new THREE.Vector3();
  148. }
  149. function handleTriangle( context, a, b, c, ua, ub, uc ) {
  150. vA = context.vertices[ a ].position;
  151. vB = context.vertices[ b ].position;
  152. vC = context.vertices[ c ].position;
  153. uvA = uv[ ua ];
  154. uvB = uv[ ub ];
  155. uvC = uv[ uc ];
  156. x1 = vB.x - vA.x;
  157. x2 = vC.x - vA.x;
  158. y1 = vB.y - vA.y;
  159. y2 = vC.y - vA.y;
  160. z1 = vB.z - vA.z;
  161. z2 = vC.z - vA.z;
  162. s1 = uvB.u - uvA.u;
  163. s2 = uvC.u - uvA.u;
  164. t1 = uvB.v - uvA.v;
  165. t2 = uvC.v - uvA.v;
  166. r = 1.0 / ( s1 * t2 - s2 * t1 );
  167. sdir.set( ( t2 * x1 - t1 * x2 ) * r,
  168. ( t2 * y1 - t1 * y2 ) * r,
  169. ( t2 * z1 - t1 * z2 ) * r );
  170. tdir.set( ( s1 * x2 - s2 * x1 ) * r,
  171. ( s1 * y2 - s2 * y1 ) * r,
  172. ( s1 * z2 - s2 * z1 ) * r );
  173. tan1[ a ].addSelf( sdir );
  174. tan1[ b ].addSelf( sdir );
  175. tan1[ c ].addSelf( sdir );
  176. tan2[ a ].addSelf( tdir );
  177. tan2[ b ].addSelf( tdir );
  178. tan2[ c ].addSelf( tdir );
  179. }
  180. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  181. face = this.faces[ f ];
  182. uv = this.faceVertexUvs[ 0 ][ f ]; // use UV layer 0 for tangents
  183. if ( face instanceof THREE.Face3 ) {
  184. handleTriangle( this, face.a, face.b, face.c, 0, 1, 2 );
  185. } else if ( face instanceof THREE.Face4 ) {
  186. handleTriangle( this, face.a, face.b, face.c, 0, 1, 2 );
  187. handleTriangle( this, face.a, face.b, face.d, 0, 1, 3 );
  188. }
  189. }
  190. var faceIndex = [ 'a', 'b', 'c', 'd' ];
  191. for ( f = 0, fl = this.faces.length; f < fl; f ++ ) {
  192. face = this.faces[ f ];
  193. for ( i = 0; i < face.vertexNormals.length; i++ ) {
  194. n.copy( face.vertexNormals[ i ] );
  195. vertexIndex = face[ faceIndex[ i ] ];
  196. t = tan1[ vertexIndex ];
  197. // Gram-Schmidt orthogonalize
  198. tmp.copy( t );
  199. tmp.subSelf( n.multiplyScalar( n.dot( t ) ) ).normalize();
  200. // Calculate handedness
  201. tmp2.cross( face.vertexNormals[ i ], t );
  202. test = tmp2.dot( tan2[ vertexIndex ] );
  203. w = (test < 0.0) ? -1.0 : 1.0;
  204. face.vertexTangents[ i ] = new THREE.Vector4( tmp.x, tmp.y, tmp.z, w );
  205. }
  206. }
  207. this.hasTangents = true;
  208. },
  209. computeBoundingBox: function () {
  210. var vertex;
  211. if ( this.vertices.length > 0 ) {
  212. this.boundingBox = { 'x': [ this.vertices[ 0 ].position.x, this.vertices[ 0 ].position.x ],
  213. 'y': [ this.vertices[ 0 ].position.y, this.vertices[ 0 ].position.y ],
  214. 'z': [ this.vertices[ 0 ].position.z, this.vertices[ 0 ].position.z ] };
  215. for ( var v = 1, vl = this.vertices.length; v < vl; v ++ ) {
  216. vertex = this.vertices[ v ];
  217. if ( vertex.position.x < this.boundingBox.x[ 0 ] ) {
  218. this.boundingBox.x[ 0 ] = vertex.position.x;
  219. } else if ( vertex.position.x > this.boundingBox.x[ 1 ] ) {
  220. this.boundingBox.x[ 1 ] = vertex.position.x;
  221. }
  222. if ( vertex.position.y < this.boundingBox.y[ 0 ] ) {
  223. this.boundingBox.y[ 0 ] = vertex.position.y;
  224. } else if ( vertex.position.y > this.boundingBox.y[ 1 ] ) {
  225. this.boundingBox.y[ 1 ] = vertex.position.y;
  226. }
  227. if ( vertex.position.z < this.boundingBox.z[ 0 ] ) {
  228. this.boundingBox.z[ 0 ] = vertex.position.z;
  229. } else if ( vertex.position.z > this.boundingBox.z[ 1 ] ) {
  230. this.boundingBox.z[ 1 ] = vertex.position.z;
  231. }
  232. }
  233. }
  234. },
  235. computeBoundingSphere: function () {
  236. // var radius = this.boundingSphere === null ? 0 : this.boundingSphere.radius;
  237. var radius = 0;
  238. for ( var v = 0, vl = this.vertices.length; v < vl; v ++ ) {
  239. radius = Math.max( radius, this.vertices[ v ].position.length() );
  240. }
  241. this.boundingSphere = { radius: radius };
  242. },
  243. computeEdgeFaces: function () {
  244. function edge_hash( a, b ) {
  245. return Math.min( a, b ) + "_" + Math.max( a, b );
  246. };
  247. function addToMap( map, hash, i ) {
  248. if ( map[ hash ] === undefined ) {
  249. map[ hash ] = { "set": {}, "array": [] };
  250. map[ hash ].set[ i ] = 1;
  251. map[ hash ].array.push( i );
  252. } else {
  253. if( map[ hash ].set[ i ] === undefined ) {
  254. map[ hash ].set[ i ] = 1;
  255. map[ hash ].array.push( i );
  256. }
  257. }
  258. };
  259. var i, il, v1, v2, j, k,
  260. face, faceIndices, faceIndex,
  261. edge,
  262. hash,
  263. vfMap = {};
  264. // construct vertex -> face map
  265. for( i = 0, il = this.faces.length; i < il; i ++ ) {
  266. face = this.faces[ i ];
  267. if ( face instanceof THREE.Face3 ) {
  268. hash = edge_hash( face.a, face.b );
  269. addToMap( vfMap, hash, i );
  270. hash = edge_hash( face.b, face.c );
  271. addToMap( vfMap, hash, i );
  272. hash = edge_hash( face.a, face.c );
  273. addToMap( vfMap, hash, i );
  274. } else if ( face instanceof THREE.Face4 ) {
  275. // in WebGLRenderer quad is tesselated
  276. // to triangles: a,b,d / b,c,d
  277. // shared edge is: b,d
  278. // should shared edge be included?
  279. // comment out if not
  280. hash = edge_hash( face.b, face.d );
  281. addToMap( vfMap, hash, i );
  282. hash = edge_hash( face.a, face.b );
  283. addToMap( vfMap, hash, i );
  284. hash = edge_hash( face.a, face.d );
  285. addToMap( vfMap, hash, i );
  286. hash = edge_hash( face.b, face.c );
  287. addToMap( vfMap, hash, i );
  288. hash = edge_hash( face.c, face.d );
  289. addToMap( vfMap, hash, i );
  290. }
  291. }
  292. // extract faces
  293. for( i = 0, il = this.edges.length; i < il; i ++ ) {
  294. edge = this.edges[ i ];
  295. v1 = edge.vertexIndices[ 0 ];
  296. v2 = edge.vertexIndices[ 1 ];
  297. edge.faceIndices = vfMap[ edge_hash( v1, v2 ) ].array;
  298. for( j = 0; j < edge.faceIndices.length; j ++ ) {
  299. faceIndex = edge.faceIndices[ j ];
  300. edge.faces.push( this.faces[ faceIndex ] );
  301. }
  302. }
  303. }
  304. };
  305. THREE.GeometryIdCounter = 0;
粤ICP备19079148号