OLD | NEW |
1 /* | 1 /* |
2 * Copyright 2010, Google Inc. | 2 * Copyright 2010, Google Inc. |
3 * All rights reserved. | 3 * All rights reserved. |
4 * | 4 * |
5 * Redistribution and use in source and binary forms, with or without | 5 * Redistribution and use in source and binary forms, with or without |
6 * modification, are permitted provided that the following conditions are | 6 * modification, are permitted provided that the following conditions are |
7 * met: | 7 * met: |
8 * | 8 * |
9 * * Redistributions of source code must retain the above copyright | 9 * * Redistributions of source code must retain the above copyright |
10 * notice, this list of conditions and the following disclaimer. | 10 * notice, this list of conditions and the following disclaimer. |
(...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
45 if (!opt_maxExtent) { | 45 if (!opt_maxExtent) { |
46 opt_maxExtent = [0, 0, 0]; | 46 opt_maxExtent = [0, 0, 0]; |
47 } | 47 } |
48 this.minExtent = [opt_minExtent[0], opt_minExtent[1], opt_minExtent[2]]; | 48 this.minExtent = [opt_minExtent[0], opt_minExtent[1], opt_minExtent[2]]; |
49 this.maxExtent = [opt_maxExtent[0], opt_maxExtent[1], opt_maxExtent[2]]; | 49 this.maxExtent = [opt_maxExtent[0], opt_maxExtent[1], opt_maxExtent[2]]; |
50 }; | 50 }; |
51 o3d.inherit('BoundingBox', 'ParamObject'); | 51 o3d.inherit('BoundingBox', 'ParamObject'); |
52 | 52 |
53 | 53 |
54 /** | 54 /** |
| 55 * Computes a list of 8 3-dimensional vectors for the corners of the box. |
| 56 * @return {!Array.<Array<numbers>>} The list of corners. |
| 57 */ |
| 58 o3d.BoundingBox.prototype.corners_ = function() { |
| 59 var result = []; |
| 60 var m = [this.minExtent, this.maxExtent]; |
| 61 for (var i = 0; i < 2; ++i) { |
| 62 for (var j = 0; j < 2; ++j) { |
| 63 for (var k = 0; k < 2; ++k) { |
| 64 result.push([m[i][0], m[j][1], m[k][2]]); |
| 65 } |
| 66 } |
| 67 } |
| 68 |
| 69 return result; |
| 70 }; |
| 71 |
| 72 |
| 73 /** |
| 74 * Computes the smallest bounding box containing all the points in the given |
| 75 * list, and either modifies the optional box passed in to match, or returns |
| 76 * that box as a new box. |
| 77 * @param {!Array.<Array<numbers>>} points A non-empty list of points. |
| 78 * @param {o3d.BoundingBox} opt_targetBox Optional box to modify instead of |
| 79 * returning a new box. |
| 80 * @private |
| 81 */ |
| 82 o3d.BoundingBox.fitBoxToPoints_ = function(points, opt_targetBox) { |
| 83 var target = opt_targetBox || new o3d.BoundingBox(); |
| 84 for (var index = 0; index < 3; ++index) { |
| 85 target.maxExtent[index] = target.minExtent[index] = points[0][index]; |
| 86 for (var i = 1; i < points.length; ++i) { |
| 87 var point = points[i]; |
| 88 target.minExtent[index] = Math.min(target.minExtent[index], point[index]); |
| 89 target.maxExtent[index] = Math.max(target.maxExtent[index], point[index]); |
| 90 } |
| 91 } |
| 92 return target; |
| 93 }; |
| 94 |
| 95 |
| 96 /** |
55 * True if this boundingbox has been initialized. | 97 * True if this boundingbox has been initialized. |
56 * @type {boolean} | 98 * @type {boolean} |
57 */ | 99 */ |
58 o3d.BoundingBox.prototype.valid_ = false; | 100 o3d.BoundingBox.prototype.valid = false; |
59 | 101 |
60 | 102 |
61 /** | 103 /** |
62 * The min extent of the box. | 104 * The min extent of the box. |
63 * @type {!o3d.math.Point3} | 105 * @type {!o3d.math.Point3} |
64 */ | 106 */ |
65 o3d.BoundingBox.prototype.minExtent = [0, 0, 0]; | 107 o3d.BoundingBox.prototype.minExtent = [0, 0, 0]; |
66 | 108 |
67 | 109 |
68 | |
69 /** | 110 /** |
70 * The max extent of the box. | 111 * The max extent of the box. |
71 * @type {!o3d.math.Point3} | 112 * @type {!o3d.math.Point3} |
72 */ | 113 */ |
73 o3d.BoundingBox.prototype.maxExtent = [0, 0, 0]; | 114 o3d.BoundingBox.prototype.maxExtent = [0, 0, 0]; |
74 | 115 |
75 | 116 |
76 | |
77 /** | 117 /** |
78 * Multiplies the bounding box by the given matrix returning a new bounding | 118 * Multiplies the bounding box by the given matrix returning a new bounding |
79 * box. | 119 * box. |
80 * @param {!o3d.math.Matrix4} matrix The matrix to multiply by. | 120 * @param {!o3d.math.Matrix4} matrix The matrix to multiply by. |
81 * @return {!o3d.BoundingBox} The new bounding box. | 121 * @return {!o3d.BoundingBox} The new bounding box. |
82 */ | 122 */ |
83 o3d.BoundingBox.prototype.mul = | 123 o3d.BoundingBox.prototype.mul = |
84 function(matrix) { | 124 function(matrix) { |
85 o3d.notImplemented(); | 125 var corners = this.corners_(); |
| 126 var new_corners = []; |
| 127 |
| 128 for (var i = 0; i < corners.length; ++i) { |
| 129 new_corners.push(o3d.Transform.transformPoint(matrix, corners[i])); |
| 130 } |
| 131 |
| 132 return o3d.BoundingBox.fitBoxToPoints_(new_corners); |
86 }; | 133 }; |
87 | 134 |
88 | 135 |
89 /** | 136 /** |
90 * Adds a bounding box to this bounding box returning a bounding box that | 137 * Adds a bounding box to this bounding box returning a bounding box that |
91 * encompases both. | 138 * encompases both. |
92 * @param {!o3d.BoundingBox} box BoundingBox to add to this BoundingBox. | 139 * @param {!o3d.BoundingBox} box BoundingBox to add to this BoundingBox. |
93 * @return {!o3d.BoundingBox} The new bounding box. | 140 * @return {!o3d.BoundingBox} The new bounding box. |
94 */ | 141 */ |
95 o3d.BoundingBox.prototype.add = | 142 o3d.BoundingBox.prototype.add = |
96 function(box) { | 143 function(box) { |
97 o3d.notImplemented(); | 144 return new o3d.BoundingBox( |
| 145 [Math.min(box.minExtent[0], this.minExtent[0]), |
| 146 Math.min(box.minExtent[1], this.minExtent[1]), |
| 147 Math.min(box.minExtent[2], this.minExtent[2])], |
| 148 [Math.max(box.maxExtent[0], this.maxExtent[0]), |
| 149 Math.max(box.maxExtent[1], this.maxExtent[1]), |
| 150 Math.max(box.maxExtent[2], this.maxExtent[2])]); |
98 }; | 151 }; |
99 | 152 |
100 | 153 |
101 /** | 154 /** |
102 * Checks if a ray defined in same coordinate system as this box intersects | 155 * Checks if a ray defined in same coordinate system as this box intersects |
103 * this bounding box. | 156 * this bounding box. |
104 * TODO(petersont): this can also take six coordinates as input. | 157 * TODO(petersont): this can also take six coordinates as input. |
105 * @param {!o3d.math.Point3} start position of start of ray in local space. | 158 * @param {!o3d.math.Point3} start position of start of ray in local space. |
106 * @param {!o3d.math.Point3} end position of end of ray in local space. | 159 * @param {!o3d.math.Point3} end position of end of ray in local space. |
107 * @return {!o3d.RayIntersectionInfo} RayIntersectionInfo. If result.value | 160 * @return {!o3d.RayIntersectionInfo} RayIntersectionInfo. If result.value |
108 * is false then something was wrong like using this function with an | 161 * is false then something was wrong like using this function with an |
109 * uninitialized bounding box. If result.intersected is true then the ray | 162 * uninitialized bounding box. If result.intersected is true then the ray |
110 * intersected the box and result.position is the exact point of | 163 * intersected the box and result.position is the exact point of |
111 * intersection. | 164 * intersection. |
112 */ | 165 */ |
113 o3d.BoundingBox.prototype.intersectRay = | 166 o3d.BoundingBox.prototype.intersectRay = |
114 function(start, end) { | 167 function(start, end) { |
115 o3d.notImplemented(); | 168 var result = new RayIntersectionInfo; |
| 169 |
| 170 if (this.valid) { |
| 171 result.valid = true; |
| 172 result.intersected = true; // True until proven false. |
| 173 |
| 174 var kNumberOfDimensions = 3; |
| 175 var kRight = 0; |
| 176 var kLeft = 1; |
| 177 var kMiddle = 2; |
| 178 |
| 179 var dir = [end[0] - start[0], end[1] - start[1], end[2] - start[2]]; |
| 180 var coord = [0, 0, 0]; |
| 181 var inside = true; |
| 182 |
| 183 var quadrant = []; |
| 184 var max_t = []; |
| 185 var candidate_plane = []; |
| 186 |
| 187 for (var i = 0; i < kNumberOfDimensions; ++i) { |
| 188 quadrant.push(0.0); |
| 189 max_t.push(0.0); |
| 190 candidate_plane.push(0,0); |
| 191 } |
| 192 |
| 193 var which_plane; |
| 194 |
| 195 // Find candidate planes; this loop can be avoided if rays cast all from |
| 196 // the eye (assumes perpsective view). |
| 197 for (var i = 0; i < kNumberOfDimensions; ++i) { |
| 198 if (start[i] < min_extent_[i]) { |
| 199 quadrant[i] = kLeft; |
| 200 candidate_plane[i] = min_extent_[i]; |
| 201 inside = false; |
| 202 } else if (start[i] > max_extent_[i]) { |
| 203 quadrant[i] = kRight; |
| 204 candidate_plane[i] = max_extent_[i]; |
| 205 inside = false; |
| 206 } else { |
| 207 quadrant[i] = kMiddle; |
| 208 } |
| 209 } |
| 210 |
| 211 // Ray origin inside bounding box. |
| 212 if (inside) { |
| 213 result.position = start; |
| 214 result.inside = true; |
| 215 } else { |
| 216 // Calculate T distances to candidate planes. |
| 217 for (var i = 0; i < kNumberOfDimensions; ++i) { |
| 218 if (quadrant[i] != kMiddle && dir[i] != 0.0) { |
| 219 max_t[i] = (candidate_plane[i] - start[i]) / dir[i]; |
| 220 } else { |
| 221 max_t[i] = -1.0; |
| 222 } |
| 223 } |
| 224 |
| 225 // Get largest of the max_t's for final choice of intersection. |
| 226 which_plane = 0; |
| 227 for (var i = 1; i < kNumberOfDimensions; ++i) { |
| 228 if (max_t[which_plane] < max_t[i]) { |
| 229 which_plane = i; |
| 230 } |
| 231 } |
| 232 |
| 233 // Check final candidate actually inside box. |
| 234 if (max_t[which_plane] < 0.0) { |
| 235 result.intersected = false; |
| 236 } else { |
| 237 for (var i = 0; i < kNumberOfDimensions; ++i) { |
| 238 if (which_plane != i) { |
| 239 coord[i] = start[i] + max_t[which_plane] * dir[i]; |
| 240 if (coord[i] < min_extent_[i] || coord[i] > max_extent_[i]) { |
| 241 result.intersected = false; |
| 242 break; |
| 243 } |
| 244 } else { |
| 245 coord[i] = candidate_plane[i]; |
| 246 } |
| 247 } |
| 248 |
| 249 // Ray hits box. |
| 250 result.position = coord; |
| 251 } |
| 252 } |
| 253 } |
| 254 |
| 255 return result; |
116 }; | 256 }; |
117 | 257 |
118 | 258 |
119 /** | 259 /** |
120 * Returns true if the bounding box is inside the frustum. | 260 * Returns true if the bounding box is inside the frustum matrix. |
| 261 * It checks all 8 corners of the bounding box against the 6 frustum planes |
| 262 * and determines whether there's at least one plane for which all 6 points lie |
| 263 * on the outside side of it. In that case it reports that the bounding box |
| 264 * is outside the frustum. Note that this is a conservative check in that |
| 265 * it in certain cases it will report that a box is in the frustum even if it |
| 266 * really isn't. However if it reports that the box is outside then it's |
| 267 * guaranteed to be outside. |
121 * @param {!o3d.math.Matrix4} matrix Matrix to transform the box from its | 268 * @param {!o3d.math.Matrix4} matrix Matrix to transform the box from its |
122 * local space to view frustum space. | 269 * local space to view frustum space. |
123 * @return {boolean} True if the box is in the frustum. | 270 * @return {boolean} True if the box is in the frustum. |
124 */ | 271 */ |
125 o3d.BoundingBox.prototype.inFrustum = | 272 o3d.BoundingBox.prototype.inFrustum = |
126 function(matrix) { | 273 function(matrix) { |
127 o3d.notImplemented(); | 274 var corners = this.corners_(); |
| 275 var bb_test = 0x3f; |
| 276 for (var i = 0; i < corners.length; ++i) { |
| 277 var corner = corners[i]; |
| 278 var p = o3d.Transform.transformPoint(matrix, corner); |
| 279 bb_test &= (((p[0] > 1.0) << 0) | |
| 280 ((p[0] < -1.0) << 1) | |
| 281 ((p[1] > 1.0) << 2) | |
| 282 ((p[1] < -1.0) << 3) | |
| 283 ((p[2] > 1.0) << 4) | |
| 284 ((p[2] < 0.0) << 5)); |
| 285 if (bb_test == 0) { |
| 286 return true; |
| 287 } |
| 288 } |
| 289 |
| 290 return (bb_test == 0); |
128 }; | 291 }; |
129 | 292 |
130 | 293 |
OLD | NEW |