Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(161)

Side by Side Diff: samples/o3d-webgl/bounding_box.js

Issue 1703014: Added culling sample. (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/o3d/
Patch Set: '' Created 10 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « samples/o3d-webgl/bitmap.js ('k') | samples/o3d-webgl/client.js » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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
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
OLDNEW
« no previous file with comments | « samples/o3d-webgl/bitmap.js ('k') | samples/o3d-webgl/client.js » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698