1 /*
2  * Copyright (c) 2009-2010 jMonkeyEngine
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met:
8  *
9  * * Redistributions of source code must retain the above copyright
10  *   notice, this list of conditions and the following disclaimer.
11  *
12  * * Redistributions in binary form must reproduce the above copyright
13  *   notice, this list of conditions and the following disclaimer in the
14  *   documentation and/or other materials provided with the distribution.
15  *
16  * * Neither the name of 'jMonkeyEngine' nor the names of its contributors
17  *   may be used to endorse or promote products derived from this software
18  *   without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
22  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
23  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
24  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
26  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
27  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
28  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
29  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
30  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31  */
32 package com.jme3.bullet;
33 
34 import com.bulletphysics.BulletGlobals;
35 import com.bulletphysics.ContactAddedCallback;
36 import com.bulletphysics.ContactDestroyedCallback;
37 import com.bulletphysics.ContactProcessedCallback;
38 import com.bulletphysics.collision.broadphase.*;
39 import com.bulletphysics.collision.dispatch.CollisionWorld.LocalConvexResult;
40 import com.bulletphysics.collision.dispatch.CollisionWorld.LocalRayResult;
41 import com.bulletphysics.collision.dispatch.*;
42 import com.bulletphysics.collision.narrowphase.ManifoldPoint;
43 import com.bulletphysics.collision.shapes.ConvexShape;
44 import com.bulletphysics.dynamics.DiscreteDynamicsWorld;
45 import com.bulletphysics.dynamics.DynamicsWorld;
46 import com.bulletphysics.dynamics.InternalTickCallback;
47 import com.bulletphysics.dynamics.RigidBody;
48 import com.bulletphysics.dynamics.constraintsolver.ConstraintSolver;
49 import com.bulletphysics.dynamics.constraintsolver.SequentialImpulseConstraintSolver;
50 import com.bulletphysics.extras.gimpact.GImpactCollisionAlgorithm;
51 import com.jme3.app.AppTask;
52 import com.jme3.asset.AssetManager;
53 import com.jme3.bullet.collision.*;
54 import com.jme3.bullet.collision.shapes.CollisionShape;
55 import com.jme3.bullet.control.PhysicsControl;
56 import com.jme3.bullet.control.RigidBodyControl;
57 import com.jme3.bullet.joints.PhysicsJoint;
58 import com.jme3.bullet.objects.PhysicsCharacter;
59 import com.jme3.bullet.objects.PhysicsGhostObject;
60 import com.jme3.bullet.objects.PhysicsRigidBody;
61 import com.jme3.bullet.objects.PhysicsVehicle;
62 import com.jme3.bullet.util.Converter;
63 import com.jme3.math.Transform;
64 import com.jme3.math.Vector3f;
65 import com.jme3.scene.Node;
66 import com.jme3.scene.Spatial;
67 import java.util.Iterator;
68 import java.util.LinkedList;
69 import java.util.List;
70 import java.util.Map;
71 import java.util.concurrent.Callable;
72 import java.util.concurrent.ConcurrentHashMap;
73 import java.util.concurrent.ConcurrentLinkedQueue;
74 import java.util.concurrent.Future;
75 import java.util.logging.Level;
76 import java.util.logging.Logger;
77 
78 /**
79  * <p>PhysicsSpace - The central jbullet-jme physics space</p>
80  * @author normenhansen
81  */
82 public class PhysicsSpace {
83 
84     public static final int AXIS_X = 0;
85     public static final int AXIS_Y = 1;
86     public static final int AXIS_Z = 2;
87     private static ThreadLocal<ConcurrentLinkedQueue<AppTask<?>>> pQueueTL =
88             new ThreadLocal<ConcurrentLinkedQueue<AppTask<?>>>() {
89 
90                 @Override
91                 protected ConcurrentLinkedQueue<AppTask<?>> initialValue() {
92                     return new ConcurrentLinkedQueue<AppTask<?>>();
93                 }
94             };
95     private ConcurrentLinkedQueue<AppTask<?>> pQueue = new ConcurrentLinkedQueue<AppTask<?>>();
96     private static ThreadLocal<PhysicsSpace> physicsSpaceTL = new ThreadLocal<PhysicsSpace>();
97     private DiscreteDynamicsWorld dynamicsWorld = null;
98     private BroadphaseInterface broadphase;
99     private BroadphaseType broadphaseType = BroadphaseType.DBVT;
100     private CollisionDispatcher dispatcher;
101     private ConstraintSolver solver;
102     private DefaultCollisionConfiguration collisionConfiguration;
103 //    private Map<GhostObject, PhysicsGhostObject> physicsGhostNodes = new ConcurrentHashMap<GhostObject, PhysicsGhostObject>();
104     private Map<RigidBody, PhysicsRigidBody> physicsNodes = new ConcurrentHashMap<RigidBody, PhysicsRigidBody>();
105     private List<PhysicsJoint> physicsJoints = new LinkedList<PhysicsJoint>();
106     private List<PhysicsCollisionListener> collisionListeners = new LinkedList<PhysicsCollisionListener>();
107     private List<PhysicsCollisionEvent> collisionEvents = new LinkedList<PhysicsCollisionEvent>();
108     private Map<Integer, PhysicsCollisionGroupListener> collisionGroupListeners = new ConcurrentHashMap<Integer, PhysicsCollisionGroupListener>();
109     private ConcurrentLinkedQueue<PhysicsTickListener> tickListeners = new ConcurrentLinkedQueue<PhysicsTickListener>();
110     private PhysicsCollisionEventFactory eventFactory = new PhysicsCollisionEventFactory();
111     private Vector3f worldMin = new Vector3f(-10000f, -10000f, -10000f);
112     private Vector3f worldMax = new Vector3f(10000f, 10000f, 10000f);
113     private float accuracy = 1f / 60f;
114     private int maxSubSteps = 4;
115     private javax.vecmath.Vector3f rayVec1 = new javax.vecmath.Vector3f();
116     private javax.vecmath.Vector3f rayVec2 = new javax.vecmath.Vector3f();
117     private com.bulletphysics.linearmath.Transform sweepTrans1 = new com.bulletphysics.linearmath.Transform(new javax.vecmath.Matrix3f());
118     private com.bulletphysics.linearmath.Transform sweepTrans2 = new com.bulletphysics.linearmath.Transform(new javax.vecmath.Matrix3f());
119     private AssetManager debugManager;
120 
121     /**
122      * Get the current PhysicsSpace <b>running on this thread</b><br/>
123      * For parallel physics, this can also be called from the OpenGL thread to receive the PhysicsSpace
124      * @return the PhysicsSpace running on this thread
125      */
getPhysicsSpace()126     public static PhysicsSpace getPhysicsSpace() {
127         return physicsSpaceTL.get();
128     }
129 
130     /**
131      * Used internally
132      * @param space
133      */
setLocalThreadPhysicsSpace(PhysicsSpace space)134     public static void setLocalThreadPhysicsSpace(PhysicsSpace space) {
135         physicsSpaceTL.set(space);
136     }
137 
PhysicsSpace()138     public PhysicsSpace() {
139         this(new Vector3f(-10000f, -10000f, -10000f), new Vector3f(10000f, 10000f, 10000f), BroadphaseType.DBVT);
140     }
141 
PhysicsSpace(BroadphaseType broadphaseType)142     public PhysicsSpace(BroadphaseType broadphaseType) {
143         this(new Vector3f(-10000f, -10000f, -10000f), new Vector3f(10000f, 10000f, 10000f), broadphaseType);
144     }
145 
PhysicsSpace(Vector3f worldMin, Vector3f worldMax)146     public PhysicsSpace(Vector3f worldMin, Vector3f worldMax) {
147         this(worldMin, worldMax, BroadphaseType.AXIS_SWEEP_3);
148     }
149 
PhysicsSpace(Vector3f worldMin, Vector3f worldMax, BroadphaseType broadphaseType)150     public PhysicsSpace(Vector3f worldMin, Vector3f worldMax, BroadphaseType broadphaseType) {
151         this.worldMin.set(worldMin);
152         this.worldMax.set(worldMax);
153         this.broadphaseType = broadphaseType;
154         create();
155     }
156 
157     /**
158      * Has to be called from the (designated) physics thread
159      */
create()160     public void create() {
161         pQueueTL.set(pQueue);
162 
163         collisionConfiguration = new DefaultCollisionConfiguration();
164         dispatcher = new CollisionDispatcher(collisionConfiguration);
165         switch (broadphaseType) {
166             case SIMPLE:
167                 broadphase = new SimpleBroadphase();
168                 break;
169             case AXIS_SWEEP_3:
170                 broadphase = new AxisSweep3(Converter.convert(worldMin), Converter.convert(worldMax));
171                 break;
172             case AXIS_SWEEP_3_32:
173                 broadphase = new AxisSweep3_32(Converter.convert(worldMin), Converter.convert(worldMax));
174                 break;
175             case DBVT:
176                 broadphase = new DbvtBroadphase();
177                 break;
178         }
179 
180         solver = new SequentialImpulseConstraintSolver();
181 
182         dynamicsWorld = new DiscreteDynamicsWorld(dispatcher, broadphase, solver, collisionConfiguration);
183         dynamicsWorld.setGravity(new javax.vecmath.Vector3f(0, -9.81f, 0));
184 
185         broadphase.getOverlappingPairCache().setInternalGhostPairCallback(new GhostPairCallback());
186         GImpactCollisionAlgorithm.registerAlgorithm(dispatcher);
187 
188         physicsSpaceTL.set(this);
189         //register filter callback for tick / collision
190         setTickCallback();
191         setContactCallbacks();
192         //register filter callback for collision groups
193         setOverlapFilterCallback();
194     }
195 
setOverlapFilterCallback()196     private void setOverlapFilterCallback() {
197         OverlapFilterCallback callback = new OverlapFilterCallback() {
198 
199             public boolean needBroadphaseCollision(BroadphaseProxy bp, BroadphaseProxy bp1) {
200                 boolean collides = (bp.collisionFilterGroup & bp1.collisionFilterMask) != 0;
201                 if (collides) {
202                     collides = (bp1.collisionFilterGroup & bp.collisionFilterMask) != 0;
203                 }
204                 if (collides) {
205                     assert (bp.clientObject instanceof com.bulletphysics.collision.dispatch.CollisionObject && bp.clientObject instanceof com.bulletphysics.collision.dispatch.CollisionObject);
206                     com.bulletphysics.collision.dispatch.CollisionObject colOb = (com.bulletphysics.collision.dispatch.CollisionObject) bp.clientObject;
207                     com.bulletphysics.collision.dispatch.CollisionObject colOb1 = (com.bulletphysics.collision.dispatch.CollisionObject) bp1.clientObject;
208                     assert (colOb.getUserPointer() != null && colOb1.getUserPointer() != null);
209                     PhysicsCollisionObject collisionObject = (PhysicsCollisionObject) colOb.getUserPointer();
210                     PhysicsCollisionObject collisionObject1 = (PhysicsCollisionObject) colOb1.getUserPointer();
211                     if ((collisionObject.getCollideWithGroups() & collisionObject1.getCollisionGroup()) > 0
212                             || (collisionObject1.getCollideWithGroups() & collisionObject.getCollisionGroup()) > 0) {
213                         PhysicsCollisionGroupListener listener = collisionGroupListeners.get(collisionObject.getCollisionGroup());
214                         PhysicsCollisionGroupListener listener1 = collisionGroupListeners.get(collisionObject1.getCollisionGroup());
215                         if (listener != null) {
216                             return listener.collide(collisionObject, collisionObject1);
217                         } else if (listener1 != null) {
218                             return listener1.collide(collisionObject, collisionObject1);
219                         }
220                         return true;
221                     } else {
222                         return false;
223                     }
224                 }
225                 return collides;
226             }
227         };
228         dynamicsWorld.getPairCache().setOverlapFilterCallback(callback);
229     }
230 
setTickCallback()231     private void setTickCallback() {
232         final PhysicsSpace space = this;
233         InternalTickCallback callback2 = new InternalTickCallback() {
234 
235             @Override
236             public void internalTick(DynamicsWorld dw, float f) {
237                 //execute task list
238                 AppTask task = pQueue.poll();
239                 task = pQueue.poll();
240                 while (task != null) {
241                     while (task.isCancelled()) {
242                         task = pQueue.poll();
243                     }
244                     try {
245                         task.invoke();
246                     } catch (Exception ex) {
247                         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.SEVERE, null, ex);
248                     }
249                     task = pQueue.poll();
250                 }
251                 for (Iterator<PhysicsTickListener> it = tickListeners.iterator(); it.hasNext();) {
252                     PhysicsTickListener physicsTickCallback = it.next();
253                     physicsTickCallback.prePhysicsTick(space, f);
254                 }
255             }
256         };
257         dynamicsWorld.setPreTickCallback(callback2);
258         InternalTickCallback callback = new InternalTickCallback() {
259 
260             @Override
261             public void internalTick(DynamicsWorld dw, float f) {
262                 for (Iterator<PhysicsTickListener> it = tickListeners.iterator(); it.hasNext();) {
263                     PhysicsTickListener physicsTickCallback = it.next();
264                     physicsTickCallback.physicsTick(space, f);
265                 }
266             }
267         };
268         dynamicsWorld.setInternalTickCallback(callback, this);
269     }
270 
setContactCallbacks()271     private void setContactCallbacks() {
272         BulletGlobals.setContactAddedCallback(new ContactAddedCallback() {
273 
274             public boolean contactAdded(ManifoldPoint cp, com.bulletphysics.collision.dispatch.CollisionObject colObj0,
275                     int partId0, int index0, com.bulletphysics.collision.dispatch.CollisionObject colObj1, int partId1,
276                     int index1) {
277                 System.out.println("contact added");
278                 return true;
279             }
280         });
281 
282         BulletGlobals.setContactProcessedCallback(new ContactProcessedCallback() {
283 
284             public boolean contactProcessed(ManifoldPoint cp, Object body0, Object body1) {
285                 if (body0 instanceof CollisionObject && body1 instanceof CollisionObject) {
286                     PhysicsCollisionObject node = null, node1 = null;
287                     CollisionObject rBody0 = (CollisionObject) body0;
288                     CollisionObject rBody1 = (CollisionObject) body1;
289                     node = (PhysicsCollisionObject) rBody0.getUserPointer();
290                     node1 = (PhysicsCollisionObject) rBody1.getUserPointer();
291                     collisionEvents.add(eventFactory.getEvent(PhysicsCollisionEvent.TYPE_PROCESSED, node, node1, cp));
292                 }
293                 return true;
294             }
295         });
296 
297         BulletGlobals.setContactDestroyedCallback(new ContactDestroyedCallback() {
298 
299             public boolean contactDestroyed(Object userPersistentData) {
300                 System.out.println("contact destroyed");
301                 return true;
302             }
303         });
304     }
305 
306     /**
307      * updates the physics space
308      * @param time the current time value
309      */
update(float time)310     public void update(float time) {
311         update(time, maxSubSteps);
312     }
313 
314     /**
315      * updates the physics space, uses maxSteps<br>
316      * @param time the current time value
317      * @param maxSteps
318      */
update(float time, int maxSteps)319     public void update(float time, int maxSteps) {
320         if (getDynamicsWorld() == null) {
321             return;
322         }
323         //step simulation
324         dynamicsWorld.stepSimulation(time, maxSteps, accuracy);
325     }
326 
distributeEvents()327     public void distributeEvents() {
328         //add collision callbacks
329         synchronized (collisionEvents) {
330             for (Iterator<PhysicsCollisionEvent> it = collisionEvents.iterator(); it.hasNext();) {
331                 PhysicsCollisionEvent physicsCollisionEvent = it.next();
332                 for (PhysicsCollisionListener listener : collisionListeners) {
333                     listener.collision(physicsCollisionEvent);
334                 }
335                 //recycle events
336                 eventFactory.recycle(physicsCollisionEvent);
337                 it.remove();
338             }
339         }
340     }
341 
enqueueOnThisThread(Callable<V> callable)342     public static <V> Future<V> enqueueOnThisThread(Callable<V> callable) {
343         AppTask<V> task = new AppTask<V>(callable);
344         System.out.println("created apptask");
345         pQueueTL.get().add(task);
346         return task;
347     }
348 
349     /**
350      * calls the callable on the next physics tick (ensuring e.g. force applying)
351      * @param <V>
352      * @param callable
353      * @return
354      */
enqueue(Callable<V> callable)355     public <V> Future<V> enqueue(Callable<V> callable) {
356         AppTask<V> task = new AppTask<V>(callable);
357         pQueue.add(task);
358         return task;
359     }
360 
361     /**
362      * adds an object to the physics space
363      * @param obj the PhysicsControl or Spatial with PhysicsControl to add
364      */
add(Object obj)365     public void add(Object obj) {
366         if (obj instanceof PhysicsControl) {
367             ((PhysicsControl) obj).setPhysicsSpace(this);
368         } else if (obj instanceof Spatial) {
369             Spatial node = (Spatial) obj;
370             PhysicsControl control = node.getControl(PhysicsControl.class);
371             control.setPhysicsSpace(this);
372         } else if (obj instanceof PhysicsCollisionObject) {
373             addCollisionObject((PhysicsCollisionObject) obj);
374         } else if (obj instanceof PhysicsJoint) {
375             addJoint((PhysicsJoint) obj);
376         } else {
377             throw (new UnsupportedOperationException("Cannot add this kind of object to the physics space."));
378         }
379     }
380 
addCollisionObject(PhysicsCollisionObject obj)381     public void addCollisionObject(PhysicsCollisionObject obj) {
382         if (obj instanceof PhysicsGhostObject) {
383             addGhostObject((PhysicsGhostObject) obj);
384         } else if (obj instanceof PhysicsRigidBody) {
385             addRigidBody((PhysicsRigidBody) obj);
386         } else if (obj instanceof PhysicsVehicle) {
387             addRigidBody((PhysicsVehicle) obj);
388         } else if (obj instanceof PhysicsCharacter) {
389             addCharacter((PhysicsCharacter) obj);
390         }
391     }
392 
393     /**
394      * removes an object from the physics space
395      * @param obj the PhysicsControl or Spatial with PhysicsControl to remove
396      */
remove(Object obj)397     public void remove(Object obj) {
398         if (obj instanceof PhysicsControl) {
399             ((PhysicsControl) obj).setPhysicsSpace(null);
400         } else if (obj instanceof Spatial) {
401             Spatial node = (Spatial) obj;
402             PhysicsControl control = node.getControl(PhysicsControl.class);
403             control.setPhysicsSpace(null);
404         } else if (obj instanceof PhysicsCollisionObject) {
405             removeCollisionObject((PhysicsCollisionObject) obj);
406         } else if (obj instanceof PhysicsJoint) {
407             removeJoint((PhysicsJoint) obj);
408         } else {
409             throw (new UnsupportedOperationException("Cannot remove this kind of object from the physics space."));
410         }
411     }
412 
removeCollisionObject(PhysicsCollisionObject obj)413     public void removeCollisionObject(PhysicsCollisionObject obj) {
414         if (obj instanceof PhysicsGhostObject) {
415             removeGhostObject((PhysicsGhostObject) obj);
416         } else if (obj instanceof PhysicsRigidBody) {
417             removeRigidBody((PhysicsRigidBody) obj);
418         } else if (obj instanceof PhysicsCharacter) {
419             removeCharacter((PhysicsCharacter) obj);
420         }
421     }
422 
423     /**
424      * adds all physics controls and joints in the given spatial node to the physics space
425      * (e.g. after loading from disk) - recursive if node
426      * @param spatial the rootnode containing the physics objects
427      */
addAll(Spatial spatial)428     public void addAll(Spatial spatial) {
429         if (spatial.getControl(RigidBodyControl.class) != null) {
430             RigidBodyControl physicsNode = spatial.getControl(RigidBodyControl.class);
431             if (!physicsNodes.containsValue(physicsNode)) {
432                 physicsNode.setPhysicsSpace(this);
433             }
434             //add joints
435             List<PhysicsJoint> joints = physicsNode.getJoints();
436             for (Iterator<PhysicsJoint> it1 = joints.iterator(); it1.hasNext();) {
437                 PhysicsJoint physicsJoint = it1.next();
438                 //add connected physicsnodes if they are not already added
439                 if (!physicsNodes.containsValue(physicsJoint.getBodyA())) {
440                     if (physicsJoint.getBodyA() instanceof PhysicsControl) {
441                         add(physicsJoint.getBodyA());
442                     } else {
443                         addRigidBody(physicsJoint.getBodyA());
444                     }
445                 }
446                 if (!physicsNodes.containsValue(physicsJoint.getBodyB())) {
447                     if (physicsJoint.getBodyA() instanceof PhysicsControl) {
448                         add(physicsJoint.getBodyB());
449                     } else {
450                         addRigidBody(physicsJoint.getBodyB());
451                     }
452                 }
453                 if (!physicsJoints.contains(physicsJoint)) {
454                     addJoint(physicsJoint);
455                 }
456             }
457         } else if (spatial.getControl(PhysicsControl.class) != null) {
458             spatial.getControl(PhysicsControl.class).setPhysicsSpace(this);
459         }
460         //recursion
461         if (spatial instanceof Node) {
462             List<Spatial> children = ((Node) spatial).getChildren();
463             for (Iterator<Spatial> it = children.iterator(); it.hasNext();) {
464                 Spatial spat = it.next();
465                 addAll(spat);
466             }
467         }
468     }
469 
470     /**
471      * Removes all physics controls and joints in the given spatial from the physics space
472      * (e.g. before saving to disk) - recursive if node
473      * @param spatial the rootnode containing the physics objects
474      */
removeAll(Spatial spatial)475     public void removeAll(Spatial spatial) {
476         if (spatial.getControl(RigidBodyControl.class) != null) {
477             RigidBodyControl physicsNode = spatial.getControl(RigidBodyControl.class);
478             if (physicsNodes.containsValue(physicsNode)) {
479                 physicsNode.setPhysicsSpace(null);
480             }
481             //remove joints
482             List<PhysicsJoint> joints = physicsNode.getJoints();
483             for (Iterator<PhysicsJoint> it1 = joints.iterator(); it1.hasNext();) {
484                 PhysicsJoint physicsJoint = it1.next();
485                 //add connected physicsnodes if they are not already added
486                 if (physicsNodes.containsValue(physicsJoint.getBodyA())) {
487                     if (physicsJoint.getBodyA() instanceof PhysicsControl) {
488                         remove(physicsJoint.getBodyA());
489                     } else {
490                         removeRigidBody(physicsJoint.getBodyA());
491                     }
492                 }
493                 if (physicsNodes.containsValue(physicsJoint.getBodyB())) {
494                     if (physicsJoint.getBodyA() instanceof PhysicsControl) {
495                         remove(physicsJoint.getBodyB());
496                     } else {
497                         removeRigidBody(physicsJoint.getBodyB());
498                     }
499                 }
500                 if (physicsJoints.contains(physicsJoint)) {
501                     removeJoint(physicsJoint);
502                 }
503             }
504         } else if (spatial.getControl(PhysicsControl.class) != null) {
505             spatial.getControl(PhysicsControl.class).setPhysicsSpace(null);
506         }
507         //recursion
508         if (spatial instanceof Node) {
509             List<Spatial> children = ((Node) spatial).getChildren();
510             for (Iterator<Spatial> it = children.iterator(); it.hasNext();) {
511                 Spatial spat = it.next();
512                 removeAll(spat);
513             }
514         }
515     }
516 
addGhostObject(PhysicsGhostObject node)517     private void addGhostObject(PhysicsGhostObject node) {
518         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Adding ghost object {0} to physics space.", node.getObjectId());
519         dynamicsWorld.addCollisionObject(node.getObjectId());
520     }
521 
removeGhostObject(PhysicsGhostObject node)522     private void removeGhostObject(PhysicsGhostObject node) {
523         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Removing ghost object {0} from physics space.", node.getObjectId());
524         dynamicsWorld.removeCollisionObject(node.getObjectId());
525     }
526 
addCharacter(PhysicsCharacter node)527     private void addCharacter(PhysicsCharacter node) {
528         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Adding character {0} to physics space.", node.getObjectId());
529 //        dynamicsWorld.addCollisionObject(node.getObjectId());
530         dynamicsWorld.addCollisionObject(node.getObjectId(), CollisionFilterGroups.CHARACTER_FILTER, (short) (CollisionFilterGroups.STATIC_FILTER | CollisionFilterGroups.DEFAULT_FILTER));
531         dynamicsWorld.addAction(node.getControllerId());
532     }
533 
removeCharacter(PhysicsCharacter node)534     private void removeCharacter(PhysicsCharacter node) {
535         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Removing character {0} from physics space.", node.getObjectId());
536         dynamicsWorld.removeAction(node.getControllerId());
537         dynamicsWorld.removeCollisionObject(node.getObjectId());
538     }
539 
addRigidBody(PhysicsRigidBody node)540     private void addRigidBody(PhysicsRigidBody node) {
541         physicsNodes.put(node.getObjectId(), node);
542 
543         //Workaround
544         //It seems that adding a Kinematic RigidBody to the dynamicWorld prevent it from being non kinematic again afterward.
545         //so we add it non kinematic, then set it kinematic again.
546         boolean kinematic = false;
547         if (node.isKinematic()) {
548             kinematic = true;
549             node.setKinematic(false);
550         }
551         dynamicsWorld.addRigidBody(node.getObjectId());
552         if (kinematic) {
553             node.setKinematic(true);
554         }
555 
556         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Adding RigidBody {0} to physics space.", node.getObjectId());
557         if (node instanceof PhysicsVehicle) {
558             Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Adding vehicle constraint {0} to physics space.", ((PhysicsVehicle) node).getVehicleId());
559             ((PhysicsVehicle) node).createVehicle(this);
560             dynamicsWorld.addVehicle(((PhysicsVehicle) node).getVehicleId());
561         }
562     }
563 
removeRigidBody(PhysicsRigidBody node)564     private void removeRigidBody(PhysicsRigidBody node) {
565         if (node instanceof PhysicsVehicle) {
566             Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Removing vehicle constraint {0} from physics space.", ((PhysicsVehicle) node).getVehicleId());
567             dynamicsWorld.removeVehicle(((PhysicsVehicle) node).getVehicleId());
568         }
569         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Removing RigidBody {0} from physics space.", node.getObjectId());
570         physicsNodes.remove(node.getObjectId());
571         dynamicsWorld.removeRigidBody(node.getObjectId());
572     }
573 
addJoint(PhysicsJoint joint)574     private void addJoint(PhysicsJoint joint) {
575         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Adding Joint {0} to physics space.", joint.getObjectId());
576         physicsJoints.add(joint);
577         dynamicsWorld.addConstraint(joint.getObjectId(), !joint.isCollisionBetweenLinkedBodys());
578     }
579 
removeJoint(PhysicsJoint joint)580     private void removeJoint(PhysicsJoint joint) {
581         Logger.getLogger(PhysicsSpace.class.getName()).log(Level.INFO, "Removing Joint {0} from physics space.", joint.getObjectId());
582         physicsJoints.remove(joint);
583         dynamicsWorld.removeConstraint(joint.getObjectId());
584     }
585 
586     /**
587      * Sets the gravity of the PhysicsSpace, set before adding physics objects!
588      * @param gravity
589      */
setGravity(Vector3f gravity)590     public void setGravity(Vector3f gravity) {
591         dynamicsWorld.setGravity(Converter.convert(gravity));
592     }
593 
594     /**
595      * applies gravity value to all objects
596      */
applyGravity()597     public void applyGravity() {
598         dynamicsWorld.applyGravity();
599     }
600 
601     /**
602      * clears forces of all objects
603      */
clearForces()604     public void clearForces() {
605         dynamicsWorld.clearForces();
606     }
607 
608     /**
609      * Adds the specified listener to the physics tick listeners.
610      * The listeners are called on each physics step, which is not necessarily
611      * each frame but is determined by the accuracy of the physics space.
612      * @param listener
613      */
addTickListener(PhysicsTickListener listener)614     public void addTickListener(PhysicsTickListener listener) {
615         tickListeners.add(listener);
616     }
617 
removeTickListener(PhysicsTickListener listener)618     public void removeTickListener(PhysicsTickListener listener) {
619         tickListeners.remove(listener);
620     }
621 
622     /**
623      * Adds a CollisionListener that will be informed about collision events
624      * @param listener the CollisionListener to add
625      */
addCollisionListener(PhysicsCollisionListener listener)626     public void addCollisionListener(PhysicsCollisionListener listener) {
627         collisionListeners.add(listener);
628     }
629 
630     /**
631      * Removes a CollisionListener from the list
632      * @param listener the CollisionListener to remove
633      */
removeCollisionListener(PhysicsCollisionListener listener)634     public void removeCollisionListener(PhysicsCollisionListener listener) {
635         collisionListeners.remove(listener);
636     }
637 
638     /**
639      * Adds a listener for a specific collision group, such a listener can disable collisions when they happen.<br>
640      * There can be only one listener per collision group.
641      * @param listener
642      * @param collisionGroup
643      */
addCollisionGroupListener(PhysicsCollisionGroupListener listener, int collisionGroup)644     public void addCollisionGroupListener(PhysicsCollisionGroupListener listener, int collisionGroup) {
645         collisionGroupListeners.put(collisionGroup, listener);
646     }
647 
removeCollisionGroupListener(int collisionGroup)648     public void removeCollisionGroupListener(int collisionGroup) {
649         collisionGroupListeners.remove(collisionGroup);
650     }
651 
652     /**
653      * Performs a ray collision test and returns the results as a list of PhysicsRayTestResults
654      */
rayTest(Vector3f from, Vector3f to)655     public List<PhysicsRayTestResult> rayTest(Vector3f from, Vector3f to) {
656         List<PhysicsRayTestResult> results = new LinkedList<PhysicsRayTestResult>();
657         dynamicsWorld.rayTest(Converter.convert(from, rayVec1), Converter.convert(to, rayVec2), new InternalRayListener(results));
658         return results;
659     }
660 
661     /**
662      * Performs a ray collision test and returns the results as a list of PhysicsRayTestResults
663      */
rayTest(Vector3f from, Vector3f to, List<PhysicsRayTestResult> results)664     public List<PhysicsRayTestResult> rayTest(Vector3f from, Vector3f to, List<PhysicsRayTestResult> results) {
665         results.clear();
666         dynamicsWorld.rayTest(Converter.convert(from, rayVec1), Converter.convert(to, rayVec2), new InternalRayListener(results));
667         return results;
668     }
669 
670     private class InternalRayListener extends CollisionWorld.RayResultCallback {
671 
672         private List<PhysicsRayTestResult> results;
673 
InternalRayListener(List<PhysicsRayTestResult> results)674         public InternalRayListener(List<PhysicsRayTestResult> results) {
675             this.results = results;
676         }
677 
678         @Override
addSingleResult(LocalRayResult lrr, boolean bln)679         public float addSingleResult(LocalRayResult lrr, boolean bln) {
680             PhysicsCollisionObject obj = (PhysicsCollisionObject) lrr.collisionObject.getUserPointer();
681             results.add(new PhysicsRayTestResult(obj, Converter.convert(lrr.hitNormalLocal), lrr.hitFraction, bln));
682             return lrr.hitFraction;
683         }
684     }
685 
686     /**
687      * Performs a sweep collision test and returns the results as a list of PhysicsSweepTestResults<br/>
688      * You have to use different Transforms for start and end (at least distance > 0.4f).
689      * SweepTest will not see a collision if it starts INSIDE an object and is moving AWAY from its center.
690      */
sweepTest(CollisionShape shape, Transform start, Transform end)691     public List<PhysicsSweepTestResult> sweepTest(CollisionShape shape, Transform start, Transform end) {
692         List<PhysicsSweepTestResult> results = new LinkedList<PhysicsSweepTestResult>();
693         if (!(shape.getCShape() instanceof ConvexShape)) {
694             Logger.getLogger(PhysicsSpace.class.getName()).log(Level.WARNING, "Trying to sweep test with incompatible mesh shape!");
695             return results;
696         }
697         dynamicsWorld.convexSweepTest((ConvexShape) shape.getCShape(), Converter.convert(start, sweepTrans1), Converter.convert(end, sweepTrans2), new InternalSweepListener(results));
698         return results;
699 
700     }
701 
702     /**
703      * Performs a sweep collision test and returns the results as a list of PhysicsSweepTestResults<br/>
704      * You have to use different Transforms for start and end (at least distance > 0.4f).
705      * SweepTest will not see a collision if it starts INSIDE an object and is moving AWAY from its center.
706      */
sweepTest(CollisionShape shape, Transform start, Transform end, List<PhysicsSweepTestResult> results)707     public List<PhysicsSweepTestResult> sweepTest(CollisionShape shape, Transform start, Transform end, List<PhysicsSweepTestResult> results) {
708         results.clear();
709         if (!(shape.getCShape() instanceof ConvexShape)) {
710             Logger.getLogger(PhysicsSpace.class.getName()).log(Level.WARNING, "Trying to sweep test with incompatible mesh shape!");
711             return results;
712         }
713         dynamicsWorld.convexSweepTest((ConvexShape) shape.getCShape(), Converter.convert(start, sweepTrans1), Converter.convert(end, sweepTrans2), new InternalSweepListener(results));
714         return results;
715     }
716 
717     private class InternalSweepListener extends CollisionWorld.ConvexResultCallback {
718 
719         private List<PhysicsSweepTestResult> results;
720 
InternalSweepListener(List<PhysicsSweepTestResult> results)721         public InternalSweepListener(List<PhysicsSweepTestResult> results) {
722             this.results = results;
723         }
724 
725         @Override
addSingleResult(LocalConvexResult lcr, boolean bln)726         public float addSingleResult(LocalConvexResult lcr, boolean bln) {
727             PhysicsCollisionObject obj = (PhysicsCollisionObject) lcr.hitCollisionObject.getUserPointer();
728             results.add(new PhysicsSweepTestResult(obj, Converter.convert(lcr.hitNormalLocal), lcr.hitFraction, bln));
729             return lcr.hitFraction;
730         }
731     }
732 
733     /**
734      * destroys the current PhysicsSpace so that a new one can be created
735      */
destroy()736     public void destroy() {
737         physicsNodes.clear();
738         physicsJoints.clear();
739 
740         dynamicsWorld.destroy();
741         dynamicsWorld = null;
742     }
743 
744     /**
745      * used internally
746      * @return the dynamicsWorld
747      */
getDynamicsWorld()748     public DynamicsWorld getDynamicsWorld() {
749         return dynamicsWorld;
750     }
751 
getBroadphaseType()752     public BroadphaseType getBroadphaseType() {
753         return broadphaseType;
754     }
755 
setBroadphaseType(BroadphaseType broadphaseType)756     public void setBroadphaseType(BroadphaseType broadphaseType) {
757         this.broadphaseType = broadphaseType;
758     }
759 
760     /**
761      * Sets the maximum amount of extra steps that will be used to step the physics
762      * when the fps is below the physics fps. Doing this maintains determinism in physics.
763      * For example a maximum number of 2 can compensate for framerates as low as 30fps
764      * when the physics has the default accuracy of 60 fps. Note that setting this
765      * value too high can make the physics drive down its own fps in case its overloaded.
766      * @param steps The maximum number of extra steps, default is 4.
767      */
setMaxSubSteps(int steps)768     public void setMaxSubSteps(int steps) {
769         maxSubSteps = steps;
770     }
771 
772     /**
773      * get the current accuracy of the physics computation
774      * @return the current accuracy
775      */
getAccuracy()776     public float getAccuracy() {
777         return accuracy;
778     }
779 
780     /**
781      * sets the accuracy of the physics computation, default=1/60s<br>
782      * @param accuracy
783      */
setAccuracy(float accuracy)784     public void setAccuracy(float accuracy) {
785         this.accuracy = accuracy;
786     }
787 
getWorldMin()788     public Vector3f getWorldMin() {
789         return worldMin;
790     }
791 
792     /**
793      * only applies for AXIS_SWEEP broadphase
794      * @param worldMin
795      */
setWorldMin(Vector3f worldMin)796     public void setWorldMin(Vector3f worldMin) {
797         this.worldMin.set(worldMin);
798     }
799 
getWorldMax()800     public Vector3f getWorldMax() {
801         return worldMax;
802     }
803 
804     /**
805      * only applies for AXIS_SWEEP broadphase
806      * @param worldMax
807      */
setWorldMax(Vector3f worldMax)808     public void setWorldMax(Vector3f worldMax) {
809         this.worldMax.set(worldMax);
810     }
811 
812     /**
813      * Enable debug display for physics
814      * @param manager AssetManager to use to create debug materials
815      */
enableDebug(AssetManager manager)816     public void enableDebug(AssetManager manager) {
817         debugManager = manager;
818     }
819 
820     /**
821      * Disable debug display
822      */
disableDebug()823     public void disableDebug() {
824         debugManager = null;
825     }
826 
getDebugManager()827     public AssetManager getDebugManager() {
828         return debugManager;
829     }
830 
831     /**
832      * interface with Broadphase types
833      */
834     public enum BroadphaseType {
835 
836         /**
837          * basic Broadphase
838          */
839         SIMPLE,
840         /**
841          * better Broadphase, needs worldBounds , max Object number = 16384
842          */
843         AXIS_SWEEP_3,
844         /**
845          * better Broadphase, needs worldBounds , max Object number = 65536
846          */
847         AXIS_SWEEP_3_32,
848         /**
849          * Broadphase allowing quicker adding/removing of physics objects
850          */
851         DBVT;
852     }
853 }
854