public:t-vien-10-3:lab_4_materials:physx
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
public:t-vien-10-3:lab_4_materials:physx [2010/10/06 23:38] – claudio | public:t-vien-10-3:lab_4_materials:physx [2024/04/29 13:33] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== Factory Scene with PhysX (unfairly only for Windows) ====== | ====== Factory Scene with PhysX (unfairly only for Windows) ====== | ||
- | Panda 1.7.0 comes with a Python exposure for PhysX, which is called Panda PhysX. Panda PhysX is a Python library you will use to access the Physx API from a Panda script. | + | The Panda 1.7.0 distribution |
- | + | - For this to work, you need to first install [[http:// | |
- | - To begin with, you need to import the Panda PhysX classes | + | - In your code, you need to import the Panda PhysX classes.< |
from panda3d.physx import PhysxManager | from panda3d.physx import PhysxManager | ||
from panda3d.physx import PhysxEnums | from panda3d.physx import PhysxEnums | ||
Line 11: | Line 11: | ||
from panda3d.physx import PhysxPlaneShapeDesc | from panda3d.physx import PhysxPlaneShapeDesc | ||
from panda3d.physx import PhysxPointOnLineJointDesc | from panda3d.physx import PhysxPointOnLineJointDesc | ||
- | </ | + | </ |
- The PhysxManager is your entry point to the PhysX engine. It is a singleton, that is a class that can have only a single global instance.< | - The PhysxManager is your entry point to the PhysX engine. It is a singleton, that is a class that can have only a single global instance.< | ||
self.physx = PhysxManager.getGlobalPtr() | self.physx = PhysxManager.getGlobalPtr() | ||
- | </ | + | </ |
- | + | ||
- | - The physics simulation take place into a PhysX scene, a sort of side invisible world where the objects have their physical incarnation. So, to begin with you want to create a PhysX Scene. Every time you want to create a general PhysX object, you need to provide a descriptor first and pass it as a parameter to a specific creation method. A descriptor is a blueprint of the object that you want to build and the creation method will return an instance of it. This fancy-looking way of instantiating classes is actually a variant of the [[http:// | + | |
# Setup the physical world | # Setup the physical world | ||
sceneDesc = PhysxSceneDesc() | sceneDesc = PhysxSceneDesc() | ||
Line 27: | Line 24: | ||
m0.setStaticFriction(0.3) | m0.setStaticFriction(0.3) | ||
m0.setDynamicFriction(0.3) | m0.setDynamicFriction(0.3) | ||
- | </ | + | </code>For each object you want to be part of the physics simulation, you have to create a PhysX Actor and set it up with a PhysX Shape (for collisions) and a PhysX Body (for rigid body dynamics). You can set both shape and body in the actor descriptor and then create the actor into the physics scene. Here is how a box actor is created (note that you still need your original Panda model of the box, that's the visible version of the box):< |
+ | # Collision shape for the box | ||
+ | shapeDesc = PhysxBoxShapeDesc() | ||
+ | shapeDesc.setDimensions(Vec3(0.3, | ||
+ | # Rigid body for the box | ||
+ | bodyDesc = PhysxBodyDesc() | ||
+ | bodyDesc.setMass(10.0) | ||
+ | # PhysX actor for the box | ||
+ | actorDesc = PhysxActorDesc() | ||
+ | actorDesc.setBody(bodyDesc) | ||
+ | actorDesc.setName(' | ||
+ | actorDesc.addShape(shapeDesc) | ||
+ | actorDesc.setGlobalPos(Point3(1, | ||
+ | self.boxActor = self.scene.createActor(actorDesc)</ | ||
+ | #--- PhysX ---- | ||
+ | taskMgr.add(self.simulate, | ||
+ | def simulate(self, | ||
+ | dt = globalClock.getDt() | ||
+ | self.scene.simulate(dt) | ||
+ | self.scene.fetchResults() # ...results fetching so to support multi-threading | ||
+ | # Set the position and rotation of the graphical box from the physical box | ||
+ | self.box.setPosQuat(self.boxActor.getGlobalPos(), | ||
+ | return task.cont | ||
+ | </ | ||
+ | - To introduce collision with the door, you will need to create an actor for the door with both collision shape and rigid body, just like you did for the box. Try using dimensions of **(1.0, 1.0, 0.1)** and mass of **500.0**. Don't forget to update the door's visual position based on the physical position in the simulation task! Your simulate task should now look like this:< | ||
+ | def simulate(self, | ||
+ | dt = globalClock.getDt() | ||
+ | self.scene.simulate(dt) | ||
+ | self.scene.fetchResults() | ||
+ | self.box.setPosQuat(self.boxActor.getGlobalPos(), | ||
+ | self.door.setPosQuat(self.doorActor.getGlobalPos(), | ||
+ | return task.cont | ||
+ | </ | ||
+ | - You need to attach the door to its door trail so that it doesn' | ||
+ | jointDesc = PhysxPointOnLineJointDesc() | ||
+ | jointDesc.setName(' | ||
+ | jointDesc.setActor(0, | ||
+ | jointDesc.setActor(1, | ||
+ | jointDesc.setGlobalAnchor(self.doorActor.getGlobalPos()) # Attach them on this point | ||
+ | jointDesc.setGlobalAxis(Vec3(1, | ||
+ | # Avoid spinning by freezing the door rotation axis | ||
+ | self.doorActor.setBodyFlag(PhysxEnums.BFFrozenRotX, | ||
+ | self.doorActor.setBodyFlag(PhysxEnums.BFFrozenRotY, | ||
+ | self.doorActor.setBodyFlag(PhysxEnums.BFFrozenRotZ, | ||
+ | # Create the joint | ||
+ | joint = self.scene.createJoint(jointDesc) | ||
+ | # Joint limits set by plane | ||
+ | joint.addLimitPlane(Vec3(-1, | ||
+ | joint.addLimitPlane(Vec3( 1, 0, 0), Point3(-1, 0, 0))</ | ||
+ | - Finally, make sure to remove all previous collision handling, since PhysX will take care of it from now on. Unfortunately, | ||
+ | - For extra awesomeness try the following: Create an instance of an AcmeBox every time you press the mouse button. Maintain a list of created boxes in your factory and iterate over that list in the simulation function to update their positions and rotations. This way you can fill the scene with tumbling boxes and be happy to mess around! |
/var/www/cadia.ru.is/wiki/data/attic/public/t-vien-10-3/lab_4_materials/physx.1286408283.txt.gz · Last modified: 2024/04/29 13:33 (external edit)