22 #ifndef SCUMM_HE_BASKETBALL_COLLISION_BBALL_COLLISION_SUPPORT_OBJ_H 23 #define SCUMM_HE_BASKETBALL_COLLISION_BBALL_COLLISION_SUPPORT_OBJ_H 27 #include "common/std/algorithm.h" 31 #define BBALL_M_PI 3.14159265358979 32 #define COPY_SIGN(x, y) ((x < 0) == (y < 0)) ? x : -x 34 #define MIN_GREATER_THAN_ZERO(a, b) ((a < 0) ? ((b < 0) ? ((a > b) ? a : b) : b) : ((b < 0) ? a : ((a > b) ? b : a))) 49 class U32Construct2D {
55 U32Construct2D() : x(0), y(0) {}
56 U32Construct2D(
Type xIn,
Type yIn) : x(xIn), y(yIn) {}
58 bool operator==(
const U32Construct2D<Type> &other)
const {
59 return ((other.x == x) && (other.y == y));
62 U32Construct2D<Type> operator-(
const U32Construct2D<Type> &other)
const {
63 U32Construct2D<Type> newConstruct;
65 newConstruct.x = x - other.x;
66 newConstruct.y = y - other.y;
71 U32Construct2D<Type>
operator+(
const U32Construct2D<Type> &other)
const {
72 U32Construct2D<Type> newConstruct;
74 newConstruct.x = x + other.x;
75 newConstruct.y = y + other.y;
80 Type operator*(
const U32Construct2D<Type> &multiplier)
const {
81 return (x * multiplier.x + y * multiplier.y);
84 Type &operator[](EDimension dimension) {
85 assert(dimension <= Y_INDEX);
86 return *(&x + dimension);
89 const Type &operator[](EDimension dimension)
const {
90 assert(dimension <= Y_INDEX);
91 return *(&x + dimension);
94 Type magnitude(
void)
const {
95 return (
Type)(sqrt(x * x + y * y));
100 class U32Construct3D {
107 U32Construct3D() : x(0), y(0), z(0) {}
108 U32Construct3D(
Type xIn,
Type yIn,
Type zIn) : x(xIn), y(yIn), z(zIn) {}
110 Type magnitude(
void)
const {
111 return (
Type)(sqrt(x * x + y * y + z * z));
114 Type xyMagnitude(
void)
const {
115 return (
Type)(sqrt(x * x + y * y));
118 bool operator==(
const U32Construct3D<Type> &other)
const {
119 return ((other.x == x) && (other.y == y) && (other.z == z));
122 bool operator!=(
const U32Construct2D<Type> &other)
const {
123 return ((other.x != x) || (other.y != y));
126 Type &operator[](EDimension dimension) {
127 assert(dimension <= Z_INDEX);
128 return *(&x + dimension);
131 const Type &operator[](EDimension dimension)
const {
132 assert(dimension <= Z_INDEX);
133 return *(&x + dimension);
136 U32Construct3D<Type>
operator+(
const U32Construct3D<Type> &other)
const {
137 U32Construct3D<Type> newConstruct;
139 newConstruct.x = x + other.x;
140 newConstruct.y = y + other.y;
141 newConstruct.z = z + other.z;
146 Type operator*(
const U32Construct3D<Type> &multiplier)
const {
147 return (x * multiplier.x + y * multiplier.y + z * multiplier.z);
150 friend U32Construct3D<Type> operator*(
const Type multiplier1[4][4], U32Construct3D<Type> multiplier2) {
151 U32Construct3D<Type> newPoint;
155 for (row = X_INDEX; row <= Z_INDEX; ++row) {
156 for (column = X_INDEX; column <= Z_INDEX; ++column) {
157 newPoint[(EDimension)row] += multiplier1[row][column] * multiplier2[(EDimension)column];
160 newPoint[(EDimension)row] += multiplier1[row][column];
163 for (column = X_INDEX; column <= Z_INDEX; ++column) {
164 h += multiplier1[row][column] * multiplier2[(EDimension)column];
167 h += multiplier1[row][column];
169 (h == 0) ? 0 : newPoint.x /= h;
170 (h == 0) ? 0 : newPoint.y /= h;
171 (h == 0) ? 0 : newPoint.z /= h;
181 class U32Point2D :
public U32Construct2D<Type> {
183 U32Point2D() : U32Construct2D<
Type>() {}
184 U32Point2D(
Type xx,
Type yy) : U32Construct2D<
Type>(xx, yy) {}
185 U32Point2D(
const U32Construct2D<Type> &other) : U32Construct2D<
Type>(other.x, other.y) {}
186 U32Point2D(
const U32Construct3D<Type> &other) : U32Construct2D<
Type>(other.x, other.y) {}
188 U32Point2D<Type> operator=(
const U32Construct3D<Type> &other) {
194 U32Construct2D<Type> operator-(
const U32Construct3D<Type> &other)
const {
195 U32Construct2D<Type> newPoint;
197 newPoint.x = this->x - other.x;
198 newPoint.y = this->y - other.y;
203 U32Construct2D<Type> operator-(
const U32Construct2D<Type> &other)
const {
204 U32Construct2D<Type> newPoint;
206 newPoint.x = this->x - other.x;
207 newPoint.y = this->y - other.y;
213 typedef U32Point2D<int> U32IntPoint2D;
214 typedef U32Point2D<float> U32FltPoint2D;
217 class U32Point3D :
public U32Construct3D<Type> {
219 U32Point3D() : U32Construct3D<
Type>() {}
220 U32Point3D(
Type xIn,
Type yIn,
Type zIn) : U32Construct3D<
Type>(xIn, yIn, zIn) {}
221 U32Point3D(
const U32Construct3D<Type> &other) : U32Construct3D<
Type>(other.x, other.y, other.z) {}
223 bool operator==(
const U32Construct2D<Type> &other)
const {
224 return ((other.x == this->x) && (other.y == this->y));
227 bool operator==(
const U32Construct3D<Type> &other)
const {
228 return ((other.x == this->x) && (other.y == this->y) && (other.z == this->z));
231 U32Construct3D<Type> operator-(
const U32Construct2D<Type> &other)
const {
232 U32Construct3D<Type> newPoint;
234 newPoint.x = this->x - other.x;
235 newPoint.y = this->y - other.y;
236 newPoint.z = this->z;
241 U32Construct3D<Type> operator-(
const U32Construct3D<Type> &other)
const {
242 U32Construct3D<Type> newPoint;
244 newPoint.x = this->x - other.x;
245 newPoint.y = this->y - other.y;
246 newPoint.z = this->z - other.z;
252 typedef U32Point3D<int> U32IntPoint3D;
253 typedef U32Point3D<float> U32FltPoint3D;
256 class U32Vector2D :
public U32Construct2D<Type> {
258 U32Vector2D() : U32Construct2D<
Type>() {}
259 U32Vector2D(
Type xx,
Type yy) : U32Construct2D<
Type>(xx, yy) {}
260 U32Vector2D(
const U32Construct2D<Type> &other) : U32Construct2D<
Type>(other.x, other.y) {}
261 U32Vector2D(
const U32Construct3D<Type> &other) : U32Construct2D<
Type>(other.x, other.y) {}
263 U32Vector2D<Type> normalize()
const {
264 Type magnitude = this->magnitude();
265 assert(magnitude > 0);
267 U32Vector2D<Type> newVector;
268 newVector.x = this->x / magnitude;
269 newVector.y = this->y / magnitude;
274 U32Vector2D<Type> operator*(
Type multiplier)
const {
275 U32Vector2D<Type> newVector;
277 newVector.x = multiplier * this->x;
278 newVector.y = multiplier * this->y;
283 Type operator*(
const U32Vector2D<Type> &multiplier)
const {
284 return (this->x * multiplier.x + this->y * multiplier.y);
287 const U32Vector2D<Type> &operator*=(
Type multiplier) {
288 *
this = *
this * multiplier;
292 const U32Vector2D<Type> &rotate(ERevDirection whichDirection,
double radians) {
293 U32Point2D<Type> newPoint;
295 if (whichDirection == kCounterClockwise) {
296 newPoint.x = this->x * cos(radians) - this->y * sin(radians);
297 newPoint.y = this->x * sin(radians) + this->y * cos(radians);
299 newPoint.x = this->x * cos(radians) + this->y * sin(radians);
300 newPoint.y = this->y * cos(radians) - this->x * sin(radians);
303 this->x = newPoint.x;
304 this->y = newPoint.y;
308 U32Construct3D<Type> cross(
const U32Vector2D<Type> &otherVector)
const {
309 U32Construct3D<Type> newVector;
313 newVector.z = this->x * otherVector.y - this->y * otherVector.x;
318 U32Vector2D<Type> transpose(
void)
const {
319 U32Vector2D<Type> newVector;
320 newVector.x = -this->y;
321 newVector.y = this->x;
326 Type distance2(
const U32Vector2D<Type> &otherVector)
const {
327 return ((this->x - otherVector.x) * (this->x - otherVector.x) +
328 (this->y - otherVector.y) * (this->y - otherVector.y));
331 Type distance(
const U32Vector2D<Type> &otherVector)
const {
332 return (
Type)(sqrt((
double)(distance2(otherVector))));
335 ERevDirection getRevDirection(
const U32Vector2D<Type> &otherVector)
const {
336 U32Construct3D<Type> vector3 = this->cross(otherVector);
339 return kCounterClockwise;
340 }
else if (vector3.z < 0) {
348 Type projectScalar(
const U32Vector2D<Type> &otherVector)
const {
349 Type projectionScalar;
351 if (otherVector.magnitude() == 0) {
352 projectionScalar = 0;
354 float otherMagnitude = otherVector.magnitude();
355 projectionScalar = (*
this * otherVector) / otherMagnitude;
358 return projectionScalar;
362 typedef U32Vector2D<int> U32IntVector2D;
363 typedef U32Vector2D<float> U32FltVector2D;
366 class U32Vector3D :
public U32Construct3D<Type> {
368 U32Vector3D() : U32Construct3D<
Type>() {}
369 U32Vector3D(
Type xx,
Type yy,
Type zz) : U32Construct3D<
Type>(xx, yy, zz) {}
370 U32Vector3D(
const U32Construct2D<Type> &other) : U32Construct3D<
Type>(other.x, other.y, 0) {}
371 U32Vector3D(
const U32Construct3D<Type> &other) : U32Construct3D<
Type>(other.x, other.y, other.z) {}
373 U32Vector3D<Type> operator-(
const U32Construct2D<Type> &other)
const {
374 U32Vector3D<Type> newVector;
376 newVector.x = this->x - other.x;
377 newVector.y = this->y - other.y;
378 newVector.z = this->z;
383 U32Vector3D<Type> operator-(
const U32Construct3D<Type> &other)
const {
384 U32Vector3D<Type> newVector;
386 newVector.x = this->x - other.x;
387 newVector.y = this->y - other.y;
388 newVector.z = this->z - other.z;
393 Type operator*(
const U32Construct3D<Type> &multiplier)
const {
394 return (this->x * multiplier.x + this->y * multiplier.y + this->z * multiplier.z);
397 U32Vector3D<Type> &
operator+=(
const U32Construct3D<Type> &adder)
const {
398 *
this = *
this + adder;
402 U32Vector3D<Type> &operator-=(
const U32Construct3D<Type> &subtractor) {
403 *
this = *
this - subtractor;
407 U32Vector3D<Type> operator*(
Type multiplier)
const {
408 U32Vector3D<Type> newVector;
410 newVector.x = multiplier * this->x;
411 newVector.y = multiplier * this->y;
412 newVector.z = multiplier * this->z;
417 const U32Vector3D<Type> &operator*=(
Type multiplier) {
418 *
this = *
this * multiplier;
422 U32Vector3D<Type> operator/(
Type divider)
const {
423 U32Vector3D<Type> newVector;
425 newVector.x = (divider == 0) ? 0 : this->x / divider;
426 newVector.y = (divider == 0) ? 0 : this->y / divider;
427 newVector.z = (divider == 0) ? 0 : this->z / divider;
432 const U32Vector3D<Type> &operator/=(
Type multiplier) {
433 *
this = *
this / multiplier;
437 U32Vector3D<Type> cross(
const U32Vector3D<Type> &otherVector)
const {
438 U32Vector3D<Type> newVector;
440 newVector.x = this->y * otherVector.z - this->z * otherVector.y;
441 newVector.y = this->z * otherVector.x - this->x * otherVector.z;
442 newVector.z = this->x * otherVector.y - this->y * otherVector.x;
447 bool operator>(
const U32Vector3D<Type> &otherVector)
const {
448 return (this->magnitude() > otherVector.magnitude());
451 bool operator<(const U32Vector3D<Type> &otherVector)
const {
452 return (this->magnitude() < otherVector.magnitude());
456 U32Vector3D<Type> projectVector(
const U32Vector3D<Type> &otherVector)
const {
457 Type projectionScalar;
459 if (otherVector.magnitude() == 0)
460 projectionScalar = 0;
462 projectionScalar = (*
this * otherVector) / (otherVector.magnitude() * otherVector.magnitude());
464 U32Vector3D<Type> newVector = otherVector * projectionScalar;
469 Type projectScalar(
const U32Vector3D<Type> &otherVector)
const {
470 Type projectionScalar;
472 if (otherVector.magnitude() == 0) {
473 projectionScalar = 0;
475 float otherMagnitude = otherVector.magnitude();
476 projectionScalar = (*
this * otherVector) / otherMagnitude;
479 return projectionScalar;
482 U32Vector3D<Type> normalize()
const {
483 Type magnitude = this->magnitude();
484 assert(magnitude > 0);
486 U32Vector3D<Type> newVector;
488 if (magnitude != 0) {
489 newVector.x = this->x / magnitude;
490 newVector.y = this->y / magnitude;
491 newVector.z = this->z / magnitude;
498 typedef U32Vector3D<int> U32IntVector3D;
499 typedef U32Vector3D<float> U32FltVector3D;
501 class U32Distance3D :
public U32Construct3D<float> {
503 U32Distance3D() : U32Construct3D<float>() {}
504 U32Distance3D(
const U32Construct2D<float> &other) : U32Construct3D<float>(other.x, other.y, 0) {}
505 U32Distance3D(
const U32Construct3D<float> &other) : U32Construct3D<float>(other.x, other.y, other.z) {}
507 U32Distance3D operator-(
int subtractor)
const {
508 U32Distance3D newDistance;
510 if (magnitude() != 0) {
511 newDistance.x = x - ((x * subtractor) / magnitude());
512 newDistance.y = y - ((y * subtractor) / magnitude());
513 newDistance.z = z - ((z * subtractor) / magnitude());
519 U32Distance3D operator-=(
int subtractor) {
520 *
this = *
this - subtractor;
527 U32FltPoint2D center;
533 U32FltPoint3D center;
537 struct U32Cylinder :
public U32Sphere {
541 struct U32BoundingBox {
543 U32IntPoint3D minPoint;
544 U32IntPoint3D maxPoint;
546 void setMinPoint(
int x,
int y) {
551 void setMaxPoint(
int x,
int y) {
556 const U32IntPoint3D &getMinPoint(
void)
const {
560 const U32IntPoint3D &getMaxPoint(
void)
const {
564 bool intersect(
const U32BoundingBox &targetBox)
const {
565 return ((((minPoint.x <= targetBox.minPoint.x) &&
566 (maxPoint.x >= targetBox.minPoint.x)) ||
568 ((minPoint.x <= targetBox.maxPoint.x) &&
569 (maxPoint.x >= targetBox.maxPoint.x)) ||
571 ((targetBox.minPoint.x <= minPoint.x) &&
572 (targetBox.maxPoint.x >= minPoint.x)) ||
574 ((targetBox.minPoint.x <= maxPoint.x) &&
575 (targetBox.maxPoint.x >= maxPoint.x)))
579 (((minPoint.y <= targetBox.minPoint.y) &&
580 (maxPoint.y >= targetBox.minPoint.y)) ||
582 ((minPoint.y <= targetBox.maxPoint.y) &&
583 (maxPoint.y >= targetBox.maxPoint.y)) ||
585 ((targetBox.minPoint.y <= minPoint.y) &&
586 (targetBox.maxPoint.y >= minPoint.y)) ||
588 ((targetBox.minPoint.y <= maxPoint.y) &&
589 (targetBox.maxPoint.y >= maxPoint.y))));
592 const U32IntPoint3D &operator[](
int point)
const {
594 return ((point == 0) ? minPoint : maxPoint);
597 U32IntPoint3D &operator[](
int point) {
599 return ((point == 0) ? minPoint : maxPoint);
602 bool isPointWithin(U32FltPoint2D point)
const {
603 return ((minPoint.x <= point.x) && (point.x <= maxPoint.x) &&
604 (minPoint.y <= point.y) && (point.y <= maxPoint.y));
610 U32FltVector3D normal;
611 float collisionEfficiency;
615 U32Plane &average(
const U32Plane &otherPlane) {
616 point.x = (point.x + otherPlane.point.x) / 2;
617 point.y = (point.y + otherPlane.point.y) / 2;
618 point.z = (point.z + otherPlane.point.z) / 2;
620 normal.x = (normal.x + otherPlane.normal.x) / 2;
621 normal.y = (normal.y + otherPlane.normal.y) / 2;
622 normal.z = (normal.z + otherPlane.normal.z) / 2;
624 float normalMag = normal.magnitude();
625 normal.x = (normalMag == 0) ? 0 : normal.x / normalMag;
626 normal.y = (normalMag == 0) ? 0 : normal.y / normalMag;
627 normal.z = (normalMag == 0) ? 0 : normal.z / normalMag;
629 collisionEfficiency = (collisionEfficiency + otherPlane.collisionEfficiency) / 2;
630 friction = (friction + otherPlane.friction) / 2;
636 point = U32FltPoint3D();
637 normal = U32FltVector3D();
638 collisionEfficiency = 0.0F;
645 U32FltPoint2D _origin;
646 U32FltVector2D _direction;
648 bool intersection(
const U32Ray2D &otherRay, U32FltPoint2D *intersection)
const {
649 float numerator = ((otherRay._origin - _origin) * otherRay._direction.transpose());
650 float denominator = _direction * otherRay._direction.transpose();
652 if ((denominator == 0) ||
653 (denominator < 0) != (numerator < 0)) {
656 float s = numerator / denominator;
659 intersection->x = _origin.x + (_direction.x * s);
660 intersection->y = _origin.y + (_direction.y * s);
665 bool nearIntersection(
const U32Circle &circle, U32FltPoint2D *intersection)
const {
666 U32FltVector2D coVector = _origin - circle.center;
667 double b = _direction.normalize() * coVector;
668 double c = (coVector * coVector) - (circle.radius * circle.radius);
669 double d = (b * b) - c;
674 double t = -b - sqrt(d);
679 *intersection = _origin + (_direction.normalize() * t);
685 bool farIntersection(
const U32Circle &circle, U32FltPoint2D *intersection)
const {
686 U32FltVector2D coVector = _origin - circle.center;
687 double b = _direction.normalize() * coVector;
688 double c = (coVector * coVector) - (circle.radius * circle.radius);
689 double d = (b * b) - c;
694 double t = -b + sqrt(d);
699 *intersection = _origin + (_direction.normalize() * t);
708 U32FltPoint3D _origin;
709 U32FltVector3D _direction;
716 #endif // SCUMM_HE_BASKETBALL_COLLISION_BBALL_COLLISION_SUPPORT_OBJ_H
DBCSString operator+(const DBCSString &x, const DBCSString &y)
RivenScriptPtr & operator+=(RivenScriptPtr &lhs, const RivenScriptPtr &rhs)