ScummVM API documentation
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
bball_collision_support_obj.h
1 /* ScummVM - Graphic Adventure Engine
2  *
3  * ScummVM is the legal property of its developers, whose names
4  * are too numerous to list here. Please refer to the COPYRIGHT
5  * file distributed with this source distribution.
6  *
7  * This program is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation, either version 3 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program. If not, see <http://www.gnu.org/licenses/>.
19  *
20  */
21 
22 #ifndef SCUMM_HE_BASKETBALL_COLLISION_BBALL_COLLISION_SUPPORT_OBJ_H
23 #define SCUMM_HE_BASKETBALL_COLLISION_BBALL_COLLISION_SUPPORT_OBJ_H
24 
25 #ifdef ENABLE_HE
26 
27 #include "common/std/algorithm.h"
28 
29 namespace Scumm {
30 
31 #define BBALL_M_PI 3.14159265358979
32 #define COPY_SIGN(x, y) ((x < 0) == (y < 0)) ? x : -x
33 
34 #define MIN_GREATER_THAN_ZERO(a, b) ((a < 0) ? ((b < 0) ? ((a > b) ? a : b) : b) : ((b < 0) ? a : ((a > b) ? b : a)))
35 
36 enum EDimension {
37  X_INDEX = 0,
38  Y_INDEX = 1,
39  Z_INDEX = 2
40 };
41 
42 enum ERevDirection {
43  kClockwise = -1,
44  kNone = 0,
45  kCounterClockwise = 1
46 };
47 
48 template<class Type>
49 class U32Construct2D {
50 
51 public:
52  Type x;
53  Type y;
54 
55  U32Construct2D() : x(0), y(0) {}
56  U32Construct2D(Type xIn, Type yIn) : x(xIn), y(yIn) {}
57 
58  bool operator==(const U32Construct2D<Type> &other) const {
59  return ((other.x == x) && (other.y == y));
60  }
61 
62  U32Construct2D<Type> operator-(const U32Construct2D<Type> &other) const {
63  U32Construct2D<Type> newConstruct;
64 
65  newConstruct.x = x - other.x;
66  newConstruct.y = y - other.y;
67 
68  return newConstruct;
69  }
70 
71  U32Construct2D<Type> operator+(const U32Construct2D<Type> &other) const {
72  U32Construct2D<Type> newConstruct;
73 
74  newConstruct.x = x + other.x;
75  newConstruct.y = y + other.y;
76 
77  return newConstruct;
78  }
79 
80  Type operator*(const U32Construct2D<Type> &multiplier) const {
81  return (x * multiplier.x + y * multiplier.y);
82  }
83 
84  Type &operator[](EDimension dimension) {
85  assert(dimension <= Y_INDEX);
86  return *(&x + dimension);
87  }
88 
89  const Type &operator[](EDimension dimension) const {
90  assert(dimension <= Y_INDEX);
91  return *(&x + dimension);
92  }
93 
94  Type magnitude(void) const {
95  return (Type)(sqrt(x * x + y * y));
96  }
97 };
98 
99 template<class Type>
100 class U32Construct3D {
101 
102 public:
103  Type x;
104  Type y;
105  Type z;
106 
107  U32Construct3D() : x(0), y(0), z(0) {}
108  U32Construct3D(Type xIn, Type yIn, Type zIn) : x(xIn), y(yIn), z(zIn) {}
109 
110  Type magnitude(void) const {
111  return (Type)(sqrt(x * x + y * y + z * z));
112  }
113 
114  Type xyMagnitude(void) const {
115  return (Type)(sqrt(x * x + y * y));
116  }
117 
118  bool operator==(const U32Construct3D<Type> &other) const {
119  return ((other.x == x) && (other.y == y) && (other.z == z));
120  }
121 
122  bool operator!=(const U32Construct2D<Type> &other) const {
123  return ((other.x != x) || (other.y != y));
124  }
125 
126  Type &operator[](EDimension dimension) {
127  assert(dimension <= Z_INDEX);
128  return *(&x + dimension);
129  }
130 
131  const Type &operator[](EDimension dimension) const {
132  assert(dimension <= Z_INDEX);
133  return *(&x + dimension);
134  }
135 
136  U32Construct3D<Type> operator+(const U32Construct3D<Type> &other) const {
137  U32Construct3D<Type> newConstruct;
138 
139  newConstruct.x = x + other.x;
140  newConstruct.y = y + other.y;
141  newConstruct.z = z + other.z;
142 
143  return newConstruct;
144  }
145 
146  Type operator*(const U32Construct3D<Type> &multiplier) const {
147  return (x * multiplier.x + y * multiplier.y + z * multiplier.z);
148  }
149 
150  friend U32Construct3D<Type> operator*(const Type multiplier1[4][4], U32Construct3D<Type> multiplier2) {
151  U32Construct3D<Type> newPoint;
152  Type h = 0;
153  int column, row;
154 
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];
158  }
159 
160  newPoint[(EDimension)row] += multiplier1[row][column];
161  }
162 
163  for (column = X_INDEX; column <= Z_INDEX; ++column) {
164  h += multiplier1[row][column] * multiplier2[(EDimension)column];
165  }
166 
167  h += multiplier1[row][column];
168 
169  (h == 0) ? 0 : newPoint.x /= h;
170  (h == 0) ? 0 : newPoint.y /= h;
171  (h == 0) ? 0 : newPoint.z /= h;
172 
173  return newPoint;
174  }
175 };
176 
177 template<class Type>
178 class U32Point3D;
179 
180 template<class Type>
181 class U32Point2D : public U32Construct2D<Type> {
182 public:
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) {} // For 3D
187 
188  U32Point2D<Type> operator=(const U32Construct3D<Type> &other) {
189  this->x = other.x;
190  this->y = other.y;
191  return *this;
192  }
193 
194  U32Construct2D<Type> operator-(const U32Construct3D<Type> &other) const {
195  U32Construct2D<Type> newPoint;
196 
197  newPoint.x = this->x - other.x;
198  newPoint.y = this->y - other.y;
199 
200  return newPoint;
201  }
202 
203  U32Construct2D<Type> operator-(const U32Construct2D<Type> &other) const {
204  U32Construct2D<Type> newPoint;
205 
206  newPoint.x = this->x - other.x;
207  newPoint.y = this->y - other.y;
208 
209  return newPoint;
210  }
211 };
212 
213 typedef U32Point2D<int> U32IntPoint2D;
214 typedef U32Point2D<float> U32FltPoint2D;
215 
216 template<class Type>
217 class U32Point3D : public U32Construct3D<Type> {
218 public:
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) {}
222 
223  bool operator==(const U32Construct2D<Type> &other) const {
224  return ((other.x == this->x) && (other.y == this->y));
225  }
226 
227  bool operator==(const U32Construct3D<Type> &other) const {
228  return ((other.x == this->x) && (other.y == this->y) && (other.z == this->z));
229  }
230 
231  U32Construct3D<Type> operator-(const U32Construct2D<Type> &other) const {
232  U32Construct3D<Type> newPoint;
233 
234  newPoint.x = this->x - other.x;
235  newPoint.y = this->y - other.y;
236  newPoint.z = this->z;
237 
238  return newPoint;
239  }
240 
241  U32Construct3D<Type> operator-(const U32Construct3D<Type> &other) const {
242  U32Construct3D<Type> newPoint;
243 
244  newPoint.x = this->x - other.x;
245  newPoint.y = this->y - other.y;
246  newPoint.z = this->z - other.z;
247 
248  return newPoint;
249  }
250 };
251 
252 typedef U32Point3D<int> U32IntPoint3D;
253 typedef U32Point3D<float> U32FltPoint3D;
254 
255 template<class Type>
256 class U32Vector2D : public U32Construct2D<Type> {
257 public:
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) {}
262 
263  U32Vector2D<Type> normalize() const {
264  Type magnitude = this->magnitude();
265  assert(magnitude > 0);
266 
267  U32Vector2D<Type> newVector;
268  newVector.x = this->x / magnitude;
269  newVector.y = this->y / magnitude;
270 
271  return newVector;
272  }
273 
274  U32Vector2D<Type> operator*(Type multiplier) const {
275  U32Vector2D<Type> newVector;
276 
277  newVector.x = multiplier * this->x;
278  newVector.y = multiplier * this->y;
279 
280  return newVector;
281  }
282 
283  Type operator*(const U32Vector2D<Type> &multiplier) const {
284  return (this->x * multiplier.x + this->y * multiplier.y);
285  }
286 
287  const U32Vector2D<Type> &operator*=(Type multiplier) {
288  *this = *this * multiplier;
289  return *this;
290  }
291 
292  const U32Vector2D<Type> &rotate(ERevDirection whichDirection, double radians) {
293  U32Point2D<Type> newPoint;
294 
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);
298  } else {
299  newPoint.x = this->x * cos(radians) + this->y * sin(radians);
300  newPoint.y = this->y * cos(radians) - this->x * sin(radians);
301  }
302 
303  this->x = newPoint.x;
304  this->y = newPoint.y;
305  return *this;
306  }
307 
308  U32Construct3D<Type> cross(const U32Vector2D<Type> &otherVector) const {
309  U32Construct3D<Type> newVector;
310 
311  newVector.x = 0;
312  newVector.y = 0;
313  newVector.z = this->x * otherVector.y - this->y * otherVector.x;
314 
315  return newVector;
316  }
317 
318  U32Vector2D<Type> transpose(void) const {
319  U32Vector2D<Type> newVector;
320  newVector.x = -this->y;
321  newVector.y = this->x;
322 
323  return newVector;
324  }
325 
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));
329  }
330 
331  Type distance(const U32Vector2D<Type> &otherVector) const {
332  return (Type)(sqrt((double)(distance2(otherVector))));
333  }
334 
335  ERevDirection getRevDirection(const U32Vector2D<Type> &otherVector) const {
336  U32Construct3D<Type> vector3 = this->cross(otherVector);
337 
338  if (vector3.z > 0) {
339  return kCounterClockwise;
340  } else if (vector3.z < 0) {
341  return kClockwise;
342  } else {
343  return kNone;
344  }
345  }
346 
347  // Project this vector onto otherVector, and return the resulting vector's magnitude.
348  Type projectScalar(const U32Vector2D<Type> &otherVector) const {
349  Type projectionScalar;
350 
351  if (otherVector.magnitude() == 0) {
352  projectionScalar = 0;
353  } else {
354  float otherMagnitude = otherVector.magnitude();
355  projectionScalar = (*this * otherVector) / otherMagnitude;
356  }
357 
358  return projectionScalar;
359  }
360 };
361 
362 typedef U32Vector2D<int> U32IntVector2D;
363 typedef U32Vector2D<float> U32FltVector2D;
364 
365 template<class Type>
366 class U32Vector3D : public U32Construct3D<Type> {
367 public:
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) {}
372 
373  U32Vector3D<Type> operator-(const U32Construct2D<Type> &other) const {
374  U32Vector3D<Type> newVector;
375 
376  newVector.x = this->x - other.x;
377  newVector.y = this->y - other.y;
378  newVector.z = this->z;
379 
380  return newVector;
381  }
382 
383  U32Vector3D<Type> operator-(const U32Construct3D<Type> &other) const {
384  U32Vector3D<Type> newVector;
385 
386  newVector.x = this->x - other.x;
387  newVector.y = this->y - other.y;
388  newVector.z = this->z - other.z;
389 
390  return newVector;
391  }
392 
393  Type operator*(const U32Construct3D<Type> &multiplier) const {
394  return (this->x * multiplier.x + this->y * multiplier.y + this->z * multiplier.z);
395  }
396 
397  U32Vector3D<Type> &operator+=(const U32Construct3D<Type> &adder) const {
398  *this = *this + adder;
399  return *this;
400  }
401 
402  U32Vector3D<Type> &operator-=(const U32Construct3D<Type> &subtractor) {
403  *this = *this - subtractor;
404  return *this;
405  }
406 
407  U32Vector3D<Type> operator*(Type multiplier) const {
408  U32Vector3D<Type> newVector;
409 
410  newVector.x = multiplier * this->x;
411  newVector.y = multiplier * this->y;
412  newVector.z = multiplier * this->z;
413 
414  return newVector;
415  }
416 
417  const U32Vector3D<Type> &operator*=(Type multiplier) {
418  *this = *this * multiplier;
419  return *this;
420  }
421 
422  U32Vector3D<Type> operator/(Type divider) const {
423  U32Vector3D<Type> newVector;
424 
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;
428 
429  return newVector;
430  }
431 
432  const U32Vector3D<Type> &operator/=(Type multiplier) {
433  *this = *this / multiplier;
434  return *this;
435  }
436 
437  U32Vector3D<Type> cross(const U32Vector3D<Type> &otherVector) const {
438  U32Vector3D<Type> newVector;
439 
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;
443 
444  return newVector;
445  }
446 
447  bool operator>(const U32Vector3D<Type> &otherVector) const {
448  return (this->magnitude() > otherVector.magnitude());
449  }
450 
451  bool operator<(const U32Vector3D<Type> &otherVector) const {
452  return (this->magnitude() < otherVector.magnitude());
453  }
454 
455  // Project this vector onto otherVector, and return the resulting vector
456  U32Vector3D<Type> projectVector(const U32Vector3D<Type> &otherVector) const {
457  Type projectionScalar;
458 
459  if (otherVector.magnitude() == 0)
460  projectionScalar = 0;
461  else
462  projectionScalar = (*this * otherVector) / (otherVector.magnitude() * otherVector.magnitude());
463 
464  U32Vector3D<Type> newVector = otherVector * projectionScalar;
465  return newVector;
466  }
467 
468  // Project this vector onto otherVector, and return the resulting vector's magnitude
469  Type projectScalar(const U32Vector3D<Type> &otherVector) const {
470  Type projectionScalar;
471 
472  if (otherVector.magnitude() == 0) {
473  projectionScalar = 0;
474  } else {
475  float otherMagnitude = otherVector.magnitude();
476  projectionScalar = (*this * otherVector) / otherMagnitude;
477  }
478 
479  return projectionScalar;
480  }
481 
482  U32Vector3D<Type> normalize() const {
483  Type magnitude = this->magnitude();
484  assert(magnitude > 0);
485 
486  U32Vector3D<Type> newVector;
487 
488  if (magnitude != 0) {
489  newVector.x = this->x / magnitude;
490  newVector.y = this->y / magnitude;
491  newVector.z = this->z / magnitude;
492  }
493 
494  return newVector;
495  }
496 };
497 
498 typedef U32Vector3D<int> U32IntVector3D;
499 typedef U32Vector3D<float> U32FltVector3D;
500 
501 class U32Distance3D : public U32Construct3D<float> {
502 public:
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) {}
506 
507  U32Distance3D operator-(int subtractor) const {
508  U32Distance3D newDistance;
509 
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());
514  }
515 
516  return newDistance;
517  }
518 
519  U32Distance3D operator-=(int subtractor) {
520  *this = *this - subtractor;
521  return *this;
522  }
523 };
524 
525 struct U32Circle {
526 
527  U32FltPoint2D center;
528  float radius;
529 };
530 
531 struct U32Sphere {
532 
533  U32FltPoint3D center;
534  float radius;
535 };
536 
537 struct U32Cylinder : public U32Sphere {
538  float height;
539 };
540 
541 struct U32BoundingBox {
542 
543  U32IntPoint3D minPoint;
544  U32IntPoint3D maxPoint;
545 
546  void setMinPoint(int x, int y) {
547  minPoint.x = x;
548  minPoint.y = y;
549  }
550 
551  void setMaxPoint(int x, int y) {
552  maxPoint.x = x;
553  maxPoint.y = y;
554  }
555 
556  const U32IntPoint3D &getMinPoint(void) const {
557  return minPoint;
558  }
559 
560  const U32IntPoint3D &getMaxPoint(void) const {
561  return maxPoint;
562  }
563 
564  bool intersect(const U32BoundingBox &targetBox) const {
565  return ((((minPoint.x <= targetBox.minPoint.x) &&
566  (maxPoint.x >= targetBox.minPoint.x)) ||
567 
568  ((minPoint.x <= targetBox.maxPoint.x) &&
569  (maxPoint.x >= targetBox.maxPoint.x)) ||
570 
571  ((targetBox.minPoint.x <= minPoint.x) &&
572  (targetBox.maxPoint.x >= minPoint.x)) ||
573 
574  ((targetBox.minPoint.x <= maxPoint.x) &&
575  (targetBox.maxPoint.x >= maxPoint.x)))
576 
577  &&
578 
579  (((minPoint.y <= targetBox.minPoint.y) &&
580  (maxPoint.y >= targetBox.minPoint.y)) ||
581 
582  ((minPoint.y <= targetBox.maxPoint.y) &&
583  (maxPoint.y >= targetBox.maxPoint.y)) ||
584 
585  ((targetBox.minPoint.y <= minPoint.y) &&
586  (targetBox.maxPoint.y >= minPoint.y)) ||
587 
588  ((targetBox.minPoint.y <= maxPoint.y) &&
589  (targetBox.maxPoint.y >= maxPoint.y))));
590  }
591 
592  const U32IntPoint3D &operator[](int point) const {
593  assert(point <= 1);
594  return ((point == 0) ? minPoint : maxPoint);
595  }
596 
597  U32IntPoint3D &operator[](int point) {
598  assert(point <= 1);
599  return ((point == 0) ? minPoint : maxPoint);
600  }
601 
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));
605  }
606 };
607 
608 struct U32Plane {
609  U32FltPoint3D point;
610  U32FltVector3D normal;
611  float collisionEfficiency;
612  float friction;
613 
614  // Average this plane with another plane
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;
619 
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;
623 
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;
628 
629  collisionEfficiency = (collisionEfficiency + otherPlane.collisionEfficiency) / 2;
630  friction = (friction + otherPlane.friction) / 2;
631 
632  return *this;
633  }
634 
635  void clear() {
636  point = U32FltPoint3D();
637  normal = U32FltVector3D();
638  collisionEfficiency = 0.0F;
639  friction = 0.0F;
640  }
641 };
642 
643 class U32Ray2D {
644 public:
645  U32FltPoint2D _origin;
646  U32FltVector2D _direction;
647 
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();
651 
652  if ((denominator == 0) ||
653  (denominator < 0) != (numerator < 0)) {
654  return false;
655  } else {
656  float s = numerator / denominator;
657  assert(s >= 0);
658 
659  intersection->x = _origin.x + (_direction.x * s);
660  intersection->y = _origin.y + (_direction.y * s);
661  return true;
662  }
663  }
664 
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;
670 
671  if (d < 0) {
672  return false;
673  } else {
674  double t = -b - sqrt(d);
675 
676  if (t < 0) {
677  return false;
678  } else {
679  *intersection = _origin + (_direction.normalize() * t);
680  return true;
681  }
682  }
683  }
684 
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;
690 
691  if (d < 0) {
692  return false;
693  } else {
694  double t = -b + sqrt(d);
695 
696  if (t < 0) {
697  return false;
698  } else {
699  *intersection = _origin + (_direction.normalize() * t);
700  return true;
701  }
702  }
703  }
704 };
705 
706 class U32Ray3D {
707 public:
708  U32FltPoint3D _origin;
709  U32FltVector3D _direction;
710 };
711 
712 } // End of namespace Scumm
713 
714 #endif // ENABLE_HE
715 
716 #endif // SCUMM_HE_BASKETBALL_COLLISION_BBALL_COLLISION_SUPPORT_OBJ_H
Type
Definition: log.h:33
DBCSString operator+(const DBCSString &x, const DBCSString &y)
RivenScriptPtr & operator+=(RivenScriptPtr &lhs, const RivenScriptPtr &rhs)
Definition: actor.h:30