Home Reference Source

cables_dev/cables/src/core/cg/cg_boundingbox.js

  1. export { BoundingBox };
  2.  
  3. /**
  4. * bounding box
  5. *
  6. * @namespace external:CGL
  7. * @param {Geometry} geometry or bounding box
  8. */
  9. class BoundingBox
  10. {
  11. constructor(geom)
  12. {
  13. this._init();
  14. this._first = true;
  15. this._wireMesh = null;
  16.  
  17. if (geom) this.apply(geom);
  18. }
  19.  
  20. _init()
  21. {
  22. this._max = [-0, -0, -0];
  23. this._min = [0, 0, 0];
  24. this._center = [0, 0, 0];
  25. this._size = [0, 0, 0];
  26. this._maxAxis = 0.0;
  27. this._first = true;
  28. }
  29.  
  30. /**
  31. * get biggest number of maxX,maxY,maxZ
  32. * @type {Number}
  33. */
  34. get maxAxis() { return this._maxAxis || 1; }
  35.  
  36. /**
  37. * size of bounding box
  38. * @type {vec3}
  39. */
  40. get size() { return this._size; }
  41.  
  42. /**
  43. * center of bounding box
  44. * @type {vec3}
  45. */
  46. get center() { return this._center; }
  47.  
  48. /**
  49. * center x
  50. * @type {Number}
  51. */
  52. get x() { return this._center[0]; }
  53.  
  54. /**
  55. * center y
  56. * @type {Number}
  57. */
  58. get y() { return this._center[1]; }
  59.  
  60. /**
  61. * center z
  62. * @type {Number}
  63. */
  64. get z() { return this._center[2]; }
  65.  
  66.  
  67. /**
  68. * minimum x
  69. * @type {Number}
  70. */
  71. get minX() { return this._min[0]; }
  72.  
  73. /**
  74. * minimum y
  75. * @type {Number}
  76. */
  77. get minY() { return this._min[1]; }
  78.  
  79. /**
  80. * minimum z
  81. * @type {Number}
  82. */
  83. get minZ() { return this._min[2]; }
  84.  
  85. /**
  86. * maximum x
  87. * @type {Number}
  88. */
  89. get maxX() { return this._max[0]; }
  90.  
  91. /**
  92. * maximum y
  93. * @type {Number}
  94. */
  95. get maxY() { return this._max[1]; }
  96.  
  97. /**
  98. * maximum z
  99. * @type {Number}
  100. */
  101. get maxZ() { return this._max[2]; }
  102.  
  103.  
  104. apply(geom, mat)
  105. {
  106. if (!geom)
  107. {
  108. // console.warn("[boundingbox] no geom/vertices", geom);
  109. return;
  110. }
  111.  
  112. if (geom instanceof BoundingBox)
  113. {
  114. const bb = geom;
  115.  
  116. this.applyPos(bb.maxX, bb.maxY, bb.maxZ);
  117. this.applyPos(bb.minX, bb.minY, bb.minZ);
  118. }
  119. else
  120. {
  121. for (let i = 0; i < geom.vertices.length; i += 3)
  122. this.applyPos(geom.vertices[i], geom.vertices[i + 1], geom.vertices[i + 2]);
  123. }
  124. this.calcCenterSize();
  125. }
  126.  
  127. /**
  128. * returns a copy of the bounding box
  129. * @function copy
  130. * @memberof BoundingBox
  131. * @instance
  132. */
  133. copy()
  134. {
  135. return new BoundingBox(this);
  136. }
  137.  
  138. get changed()
  139. {
  140. return !(this._max[0] == -Number.MAX_VALUE && this._max[1] == -Number.MAX_VALUE && this._max[2] == -Number.MAX_VALUE);
  141. }
  142.  
  143. applyPos(x, y, z)
  144. {
  145. if (x == Number.MAX_VALUE || x == -Number.MAX_VALUE ||
  146. y == Number.MAX_VALUE || y == -Number.MAX_VALUE ||
  147. z == Number.MAX_VALUE || z == -Number.MAX_VALUE) return;
  148.  
  149. if (!CABLES.UTILS.isNumeric(x) || !CABLES.UTILS.isNumeric(y) || !CABLES.UTILS.isNumeric(z)) return;
  150.  
  151. if (this._first)
  152. {
  153. this._max[0] = x;
  154. this._max[1] = y;
  155. this._max[2] = z;
  156.  
  157. this._min[0] = x;
  158. this._min[1] = y;
  159. this._min[2] = z;
  160. this._first = false;
  161. return;
  162. }
  163.  
  164. this._max[0] = Math.max(this._max[0], x);
  165. this._max[1] = Math.max(this._max[1], y);
  166. this._max[2] = Math.max(this._max[2], z);
  167.  
  168. this._min[0] = Math.min(this._min[0], x);
  169. this._min[1] = Math.min(this._min[1], y);
  170. this._min[2] = Math.min(this._min[2], z);
  171. }
  172.  
  173. calcCenterSize()
  174. {
  175. if (this._first) return;
  176.  
  177.  
  178. this._size[0] = this._max[0] - this._min[0];
  179. this._size[1] = this._max[1] - this._min[1];
  180. this._size[2] = this._max[2] - this._min[2];
  181.  
  182. this._center[0] = (this._min[0] + this._max[0]) / 2;
  183. this._center[1] = (this._min[1] + this._max[1]) / 2;
  184. this._center[2] = (this._min[2] + this._max[2]) / 2;
  185.  
  186. this._maxAxis = Math.max(this._size[2], Math.max(this._size[0], this._size[1]));
  187. }
  188.  
  189. mulMat4(m)
  190. {
  191. if (this._first)
  192. {
  193. this._max[0] = 0;
  194. this._max[1] = 0;
  195. this._max[2] = 0;
  196.  
  197. this._min[0] = 0;
  198. this._min[1] = 0;
  199. this._min[2] = 0;
  200. this._first = false;
  201. }
  202. vec3.transformMat4(this._max, this._max, m);
  203. vec3.transformMat4(this._min, this._min, m);
  204. this.calcCenterSize();
  205. }
  206.  
  207. render(cgl, shader, op)
  208. {
  209. if (!this._wireMesh) this._wireMesh = new CGL.WireCube(cgl);
  210.  
  211. cgl.pushModelMatrix();
  212. mat4.translate(cgl.mMatrix, cgl.mMatrix, this._center);
  213.  
  214. if (CABLES.UI && op)
  215. {
  216. CABLES.UI.OverlayMeshes.drawCube(op, this._size[0] / 2, this._size[1] / 2, this._size[2] / 2);
  217. }
  218.  
  219. cgl.popModelMatrix();
  220. }
  221. }