add test for issue #1125 (heightfield shapes don't match TerrainQuad)
parent
5586c28c95
commit
df02333cd2
@ -0,0 +1,229 @@ |
|||||||
|
/* |
||||||
|
* Copyright (c) 2019 jMonkeyEngine |
||||||
|
* All rights reserved. |
||||||
|
* |
||||||
|
* Redistribution and use in source and binary forms, with or without |
||||||
|
* modification, are permitted provided that the following conditions are |
||||||
|
* met: |
||||||
|
* |
||||||
|
* * Redistributions of source code must retain the above copyright |
||||||
|
* notice, this list of conditions and the following disclaimer. |
||||||
|
* |
||||||
|
* * Redistributions in binary form must reproduce the above copyright |
||||||
|
* notice, this list of conditions and the following disclaimer in the |
||||||
|
* documentation and/or other materials provided with the distribution. |
||||||
|
* |
||||||
|
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
||||||
|
* may be used to endorse or promote products derived from this software |
||||||
|
* without specific prior written permission. |
||||||
|
* |
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
||||||
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
||||||
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
||||||
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
||||||
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
||||||
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
||||||
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
||||||
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
||||||
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||||
|
*/ |
||||||
|
package jme3test.bullet; |
||||||
|
|
||||||
|
import com.jme3.app.SimpleApplication; |
||||||
|
import com.jme3.bullet.BulletAppState; |
||||||
|
import com.jme3.bullet.PhysicsSpace; |
||||||
|
import com.jme3.bullet.collision.shapes.CollisionShape; |
||||||
|
import com.jme3.bullet.control.RigidBodyControl; |
||||||
|
import com.jme3.bullet.util.CollisionShapeFactory; |
||||||
|
import com.jme3.font.BitmapText; |
||||||
|
import com.jme3.material.Material; |
||||||
|
import com.jme3.material.RenderState; |
||||||
|
import com.jme3.math.ColorRGBA; |
||||||
|
import com.jme3.math.Quaternion; |
||||||
|
import com.jme3.math.Vector3f; |
||||||
|
import com.jme3.terrain.geomipmap.TerrainQuad; |
||||||
|
import java.util.logging.Logger; |
||||||
|
|
||||||
|
/** |
||||||
|
* Test case for JME issue #1125: heightfield collision shapes don't match |
||||||
|
* TerrainQuad. |
||||||
|
* <p> |
||||||
|
* If successful, just one set of grid diagonals will be visible. If |
||||||
|
* unsuccessful, you'll see both green diagonals (the TerrainQuad) and |
||||||
|
* perpendicular blue diagonals (physics debug). |
||||||
|
* <p> |
||||||
|
* Use this test with jme3-bullet only; it can yield false success with |
||||||
|
* jme3-jbullet due to JME issue #1129. |
||||||
|
* |
||||||
|
* @author Stephen Gold sgold@sonic.net |
||||||
|
*/ |
||||||
|
public class TestIssue1125 extends SimpleApplication { |
||||||
|
// *************************************************************************
|
||||||
|
// constants and loggers
|
||||||
|
|
||||||
|
/** |
||||||
|
* message logger for this class
|
||||||
|
*/ |
||||||
|
final public static Logger logger |
||||||
|
= Logger.getLogger(TestIssue1125.class.getName()); |
||||||
|
// *************************************************************************
|
||||||
|
// fields
|
||||||
|
|
||||||
|
/** |
||||||
|
* height array for a small heightfield |
||||||
|
*/ |
||||||
|
final private float[] nineHeights = new float[9]; |
||||||
|
/** |
||||||
|
* green wireframe material for the TerrainQuad |
||||||
|
*/ |
||||||
|
private Material quadMaterial; |
||||||
|
/** |
||||||
|
* space for physics simulation |
||||||
|
*/ |
||||||
|
private PhysicsSpace physicsSpace; |
||||||
|
// *************************************************************************
|
||||||
|
// new methods exposed
|
||||||
|
|
||||||
|
/** |
||||||
|
* Main entry point for the TestIssue1125 application. |
||||||
|
* |
||||||
|
* @param ignored array of command-line arguments (not null) |
||||||
|
*/ |
||||||
|
public static void main(String[] ignored) { |
||||||
|
new TestIssue1125().start(); |
||||||
|
} |
||||||
|
// *************************************************************************
|
||||||
|
// SimpleApplication methods
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize this application. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public void simpleInitApp() { |
||||||
|
configureCamera(); |
||||||
|
configureMaterials(); |
||||||
|
viewPort.setBackgroundColor(new ColorRGBA(0.5f, 0.2f, 0.2f, 1f)); |
||||||
|
configurePhysics(); |
||||||
|
initializeHeightData(); |
||||||
|
addTerrain(); |
||||||
|
showHints(); |
||||||
|
} |
||||||
|
// *************************************************************************
|
||||||
|
// private methods
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add 3x3 terrain to the scene and the PhysicsSpace. |
||||||
|
*/ |
||||||
|
private void addTerrain() { |
||||||
|
int patchSize = 3; |
||||||
|
int mapSize = 3; |
||||||
|
TerrainQuad quad |
||||||
|
= new TerrainQuad("terrain", patchSize, mapSize, nineHeights); |
||||||
|
rootNode.attachChild(quad); |
||||||
|
quad.setMaterial(quadMaterial); |
||||||
|
|
||||||
|
CollisionShape shape = CollisionShapeFactory.createMeshShape(quad); |
||||||
|
float massForStatic = 0f; |
||||||
|
RigidBodyControl rbc = new RigidBodyControl(shape, massForStatic); |
||||||
|
rbc.setPhysicsSpace(physicsSpace); |
||||||
|
quad.addControl(rbc); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Configure the camera during startup. |
||||||
|
*/ |
||||||
|
private void configureCamera() { |
||||||
|
float fHeight = cam.getFrustumTop() - cam.getFrustumBottom(); |
||||||
|
float fWidth = cam.getFrustumRight() - cam.getFrustumLeft(); |
||||||
|
float fAspect = fWidth / fHeight; |
||||||
|
float yDegrees = 45f; |
||||||
|
float near = 0.02f; |
||||||
|
float far = 20f; |
||||||
|
cam.setFrustumPerspective(yDegrees, fAspect, near, far); |
||||||
|
|
||||||
|
flyCam.setMoveSpeed(5f); |
||||||
|
|
||||||
|
cam.setLocation(new Vector3f(2f, 4.7f, 0.4f)); |
||||||
|
cam.setRotation(new Quaternion(0.348f, -0.64f, 0.4f, 0.556f)); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Configure materials during startup. |
||||||
|
*/ |
||||||
|
private void configureMaterials() { |
||||||
|
quadMaterial = new Material(assetManager, |
||||||
|
"Common/MatDefs/Misc/Unshaded.j3md"); |
||||||
|
quadMaterial.setColor("Color", ColorRGBA.Green.clone()); |
||||||
|
RenderState ars = quadMaterial.getAdditionalRenderState(); |
||||||
|
ars.setWireframe(true); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Configure physics during startup. |
||||||
|
*/ |
||||||
|
private void configurePhysics() { |
||||||
|
BulletAppState bulletAppState = new BulletAppState(); |
||||||
|
stateManager.attach(bulletAppState); |
||||||
|
bulletAppState.setDebugEnabled(true); |
||||||
|
physicsSpace = bulletAppState.getPhysicsSpace(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize the height data during startup. |
||||||
|
*/ |
||||||
|
private void initializeHeightData() { |
||||||
|
nineHeights[0] = 1f; |
||||||
|
nineHeights[1] = 0f; |
||||||
|
nineHeights[2] = 1f; |
||||||
|
nineHeights[3] = 0f; |
||||||
|
nineHeights[4] = 0.5f; |
||||||
|
nineHeights[5] = 0f; |
||||||
|
nineHeights[6] = 1f; |
||||||
|
nineHeights[7] = 0f; |
||||||
|
nineHeights[8] = 1f; |
||||||
|
} |
||||||
|
|
||||||
|
private void showHints() { |
||||||
|
guiFont = assetManager.loadFont("Interface/Fonts/Default.fnt"); |
||||||
|
int numLines = 3; |
||||||
|
BitmapText lines[] = new BitmapText[numLines]; |
||||||
|
for (int i = 0; i < numLines; ++i) { |
||||||
|
lines[i] = new BitmapText(guiFont); |
||||||
|
} |
||||||
|
|
||||||
|
String p = "Test for jMonkeyEngine issue #1125"; |
||||||
|
if (isNativeBullet()) { |
||||||
|
lines[0].setText(p + " with native Bullet"); |
||||||
|
} else { |
||||||
|
lines[0].setText(p + " with JBullet (may yield false success)"); |
||||||
|
} |
||||||
|
lines[1].setText("Use W/A/S/D/Q/Z/arrow keys to move the camera."); |
||||||
|
lines[2].setText("F5: render stats, C: camera pos, M: mem stats"); |
||||||
|
|
||||||
|
float textHeight = guiFont.getCharSet().getLineHeight(); |
||||||
|
float viewHeight = cam.getHeight(); |
||||||
|
float viewWidth = cam.getWidth(); |
||||||
|
for (int i = 0; i < numLines; ++i) { |
||||||
|
float left = Math.round((viewWidth - lines[i].getLineWidth()) / 2f); |
||||||
|
float top = viewHeight - i * textHeight; |
||||||
|
lines[i].setLocalTranslation(left, top, 0f); |
||||||
|
guiNode.attachChild(lines[i]); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Determine which physics library is in use. |
||||||
|
* |
||||||
|
* @return true for C++ Bullet, false for JBullet (jme3-jbullet) |
||||||
|
*/ |
||||||
|
private boolean isNativeBullet() { |
||||||
|
try { |
||||||
|
Class clazz = Class.forName("com.jme3.bullet.util.NativeMeshUtil"); |
||||||
|
return clazz != null; |
||||||
|
} catch (ClassNotFoundException exception) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue