Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/environment2/src/external/bullet/BulletCollision/CollisionDispatch/SphereTriangleDetector.cpp @ 8748

Last change on this file since 8748 was 8351, checked in by rgrieder, 14 years ago

Merged kicklib2 branch back to trunk (includes former branches ois_update, mac_osx and kicklib).

Notes for updating

Linux:
You don't need an extra package for CEGUILua and Tolua, it's already shipped with CEGUI.
However you do need to make sure that the OgreRenderer is installed too with CEGUI 0.7 (may be a separate package).
Also, Orxonox now recognises if you install the CgProgramManager (a separate package available on newer Ubuntu on Debian systems).

Windows:
Download the new dependency packages versioned 6.0 and use these. If you have problems with that or if you don't like the in game console problem mentioned below, you can download the new 4.3 version of the packages (only available for Visual Studio 2005/2008).

Key new features:

  • *Support for Mac OS X*
  • Visual Studio 2010 support
  • Bullet library update to 2.77
  • OIS library update to 1.3
  • Support for CEGUI 0.7 —> Support for Arch Linux and even SuSE
  • Improved install target
  • Compiles now with GCC 4.6
  • Ogre Cg Shader plugin activated for Linux if available
  • And of course lots of bug fixes

There are also some regressions:

  • No support for CEGUI 0.5, Ogre 1.4 and boost 1.35 - 1.39 any more
  • In game console is not working in main menu for CEGUI 0.7
  • Tolua (just the C lib, not the application) and CEGUILua libraries are no longer in our repository. —> You will need to get these as well when compiling Orxonox
  • And of course lots of new bugs we don't yet know about
  • Property svn:eol-style set to native
File size: 6.3 KB
Line 
1/*
2Bullet Continuous Collision Detection and Physics Library
3Copyright (c) 2003-2006 Erwin Coumans  http://continuousphysics.com/Bullet/
4
5This software is provided 'as-is', without any express or implied warranty.
6In no event will the authors be held liable for any damages arising from the use of this software.
7Permission is granted to anyone to use this software for any purpose,
8including commercial applications, and to alter it and redistribute it freely,
9subject to the following restrictions:
10
111. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
122. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
133. This notice may not be removed or altered from any source distribution.
14*/
15
16#include "LinearMath/btScalar.h"
17#include "SphereTriangleDetector.h"
18#include "BulletCollision/CollisionShapes/btTriangleShape.h"
19#include "BulletCollision/CollisionShapes/btSphereShape.h"
20
21
22SphereTriangleDetector::SphereTriangleDetector(btSphereShape* sphere,btTriangleShape* triangle,btScalar contactBreakingThreshold)
23:m_sphere(sphere),
24m_triangle(triangle),
25m_contactBreakingThreshold(contactBreakingThreshold)
26{
27
28}
29
30void    SphereTriangleDetector::getClosestPoints(const ClosestPointInput& input,Result& output,class btIDebugDraw* debugDraw,bool swapResults)
31{
32
33        (void)debugDraw;
34        const btTransform& transformA = input.m_transformA;
35        const btTransform& transformB = input.m_transformB;
36
37        btVector3 point,normal;
38        btScalar timeOfImpact = btScalar(1.);
39        btScalar depth = btScalar(0.);
40//      output.m_distance = btScalar(BT_LARGE_FLOAT);
41        //move sphere into triangle space
42        btTransform     sphereInTr = transformB.inverseTimes(transformA);
43
44        if (collide(sphereInTr.getOrigin(),point,normal,depth,timeOfImpact,m_contactBreakingThreshold))
45        {
46                if (swapResults)
47                {
48                        btVector3 normalOnB = transformB.getBasis()*normal;
49                        btVector3 normalOnA = -normalOnB;
50                        btVector3 pointOnA = transformB*point+normalOnB*depth;
51                        output.addContactPoint(normalOnA,pointOnA,depth);
52                } else
53                {
54                        output.addContactPoint(transformB.getBasis()*normal,transformB*point,depth);
55                }
56        }
57
58}
59
60#define MAX_OVERLAP btScalar(0.)
61
62
63
64// See also geometrictools.com
65// Basic idea: D = |p - (lo + t0*lv)| where t0 = lv . (p - lo) / lv . lv
66btScalar SegmentSqrDistance(const btVector3& from, const btVector3& to,const btVector3 &p, btVector3 &nearest);
67
68btScalar SegmentSqrDistance(const btVector3& from, const btVector3& to,const btVector3 &p, btVector3 &nearest) {
69        btVector3 diff = p - from;
70        btVector3 v = to - from;
71        btScalar t = v.dot(diff);
72       
73        if (t > 0) {
74                btScalar dotVV = v.dot(v);
75                if (t < dotVV) {
76                        t /= dotVV;
77                        diff -= t*v;
78                } else {
79                        t = 1;
80                        diff -= v;
81                }
82        } else
83                t = 0;
84
85        nearest = from + t*v;
86        return diff.dot(diff); 
87}
88
89bool SphereTriangleDetector::facecontains(const btVector3 &p,const btVector3* vertices,btVector3& normal)  {
90        btVector3 lp(p);
91        btVector3 lnormal(normal);
92       
93        return pointInTriangle(vertices, lnormal, &lp);
94}
95
96///combined discrete/continuous sphere-triangle
97bool SphereTriangleDetector::collide(const btVector3& sphereCenter,btVector3 &point, btVector3& resultNormal, btScalar& depth, btScalar &timeOfImpact, btScalar contactBreakingThreshold)
98{
99
100        const btVector3* vertices = &m_triangle->getVertexPtr(0);
101        const btVector3& c = sphereCenter;
102        btScalar r = m_sphere->getRadius();
103
104        btVector3 delta (0,0,0);
105
106        btVector3 normal = (vertices[1]-vertices[0]).cross(vertices[2]-vertices[0]);
107        normal.normalize();
108        btVector3 p1ToCentre = c - vertices[0];
109        btScalar distanceFromPlane = p1ToCentre.dot(normal);
110
111        if (distanceFromPlane < btScalar(0.))
112        {
113                //triangle facing the other way
114       
115                distanceFromPlane *= btScalar(-1.);
116                normal *= btScalar(-1.);
117        }
118
119        btScalar contactMargin = contactBreakingThreshold;
120        bool isInsideContactPlane = distanceFromPlane < r + contactMargin;
121        bool isInsideShellPlane = distanceFromPlane < r;
122       
123        btScalar deltaDotNormal = delta.dot(normal);
124        if (!isInsideShellPlane && deltaDotNormal >= btScalar(0.0))
125                return false;
126
127        // Check for contact / intersection
128        bool hasContact = false;
129        btVector3 contactPoint;
130        if (isInsideContactPlane) {
131                if (facecontains(c,vertices,normal)) {
132                        // Inside the contact wedge - touches a point on the shell plane
133                        hasContact = true;
134                        contactPoint = c - normal*distanceFromPlane;
135                } else {
136                        // Could be inside one of the contact capsules
137                        btScalar contactCapsuleRadiusSqr = (r + contactMargin) * (r + contactMargin);
138                        btVector3 nearestOnEdge;
139                        for (int i = 0; i < m_triangle->getNumEdges(); i++) {
140                               
141                                btVector3 pa;
142                                btVector3 pb;
143                               
144                                m_triangle->getEdge(i,pa,pb);
145
146                                btScalar distanceSqr = SegmentSqrDistance(pa,pb,c, nearestOnEdge);
147                                if (distanceSqr < contactCapsuleRadiusSqr) {
148                                        // Yep, we're inside a capsule
149                                        hasContact = true;
150                                        contactPoint = nearestOnEdge;
151                                }
152                               
153                        }
154                }
155        }
156
157        if (hasContact) {
158                btVector3 contactToCentre = c - contactPoint;
159                btScalar distanceSqr = contactToCentre.length2();
160                if (distanceSqr < (r - MAX_OVERLAP)*(r - MAX_OVERLAP)) {
161                        btScalar distance = btSqrt(distanceSqr);
162                        resultNormal = contactToCentre;
163                        resultNormal.normalize();
164                        point = contactPoint;
165                        depth = -(r-distance);
166                        return true;
167                }
168
169                if (delta.dot(contactToCentre) >= btScalar(0.0)) 
170                        return false;
171               
172                // Moving towards the contact point -> collision
173                point = contactPoint;
174                timeOfImpact = btScalar(0.0);
175                return true;
176        }
177       
178        return false;
179}
180
181
182bool SphereTriangleDetector::pointInTriangle(const btVector3 vertices[], const btVector3 &normal, btVector3 *p )
183{
184        const btVector3* p1 = &vertices[0];
185        const btVector3* p2 = &vertices[1];
186        const btVector3* p3 = &vertices[2];
187
188        btVector3 edge1( *p2 - *p1 );
189        btVector3 edge2( *p3 - *p2 );
190        btVector3 edge3( *p1 - *p3 );
191
192        btVector3 p1_to_p( *p - *p1 );
193        btVector3 p2_to_p( *p - *p2 );
194        btVector3 p3_to_p( *p - *p3 );
195
196        btVector3 edge1_normal( edge1.cross(normal));
197        btVector3 edge2_normal( edge2.cross(normal));
198        btVector3 edge3_normal( edge3.cross(normal));
199       
200        btScalar r1, r2, r3;
201        r1 = edge1_normal.dot( p1_to_p );
202        r2 = edge2_normal.dot( p2_to_p );
203        r3 = edge3_normal.dot( p3_to_p );
204        if ( ( r1 > 0 && r2 > 0 && r3 > 0 ) ||
205             ( r1 <= 0 && r2 <= 0 && r3 <= 0 ) )
206                return true;
207        return false;
208
209}
Note: See TracBrowser for help on using the repository browser.