|
|
@ -1,39 +1,35 @@ |
|
|
|
/* |
|
|
|
/* |
|
|
|
* Copyright (c) 2009-2010 jMonkeyEngine |
|
|
|
* Copyright (c) 2009-2010 jMonkeyEngine All rights reserved. |
|
|
|
* All rights reserved. |
|
|
|
* <p/> |
|
|
|
* |
|
|
|
|
|
|
|
* Redistribution and use in source and binary forms, with or without |
|
|
|
* Redistribution and use in source and binary forms, with or without |
|
|
|
* modification, are permitted provided that the following conditions are |
|
|
|
* modification, are permitted provided that the following conditions are met: |
|
|
|
* met: |
|
|
|
* |
|
|
|
* |
|
|
|
* * Redistributions of source code must retain the above copyright notice, |
|
|
|
* * Redistributions of source code must retain the above copyright |
|
|
|
* this list of conditions and the following disclaimer. |
|
|
|
* notice, this list of conditions and the following disclaimer. |
|
|
|
* <p/> |
|
|
|
* |
|
|
|
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright |
|
|
|
* * Redistributions in binary form must reproduce the above copyright |
|
|
|
* notice, this list of conditions and the following disclaimer in the |
|
|
|
* notice, this list of conditions and the following disclaimer in the |
|
|
|
* documentation and/or other materials provided with the distribution. |
|
|
|
* documentation and/or other materials provided with the distribution. |
|
|
|
* |
|
|
|
* <p/> |
|
|
|
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
|
|
|
* * Neither the name of 'jMonkeyEngine' nor the names of its contributors |
|
|
|
* may be used to endorse or promote products derived from this software |
|
|
|
* may be used to endorse or promote products derived from this software |
|
|
|
* without specific prior written permission. |
|
|
|
* without specific prior written permission. |
|
|
|
* |
|
|
|
* <p/> |
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED |
|
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
|
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR |
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE |
|
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
|
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
|
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
|
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
|
|
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
|
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
|
|
* POSSIBILITY OF SUCH DAMAGE. |
|
|
|
*/ |
|
|
|
*/ |
|
|
|
package jme3test.input; |
|
|
|
package jme3test.input; |
|
|
|
|
|
|
|
|
|
|
|
import com.jme3.app.SettingsDialog; |
|
|
|
|
|
|
|
import com.jme3.app.SimpleApplication; |
|
|
|
import com.jme3.app.SimpleApplication; |
|
|
|
import com.jme3.input.ChaseCamera; |
|
|
|
|
|
|
|
import com.jme3.input.KeyInput; |
|
|
|
import com.jme3.input.KeyInput; |
|
|
|
import com.jme3.input.MouseInput; |
|
|
|
import com.jme3.input.MouseInput; |
|
|
|
import com.jme3.input.controls.ActionListener; |
|
|
|
import com.jme3.input.controls.ActionListener; |
|
|
@ -51,111 +47,112 @@ import com.jme3.scene.Node; |
|
|
|
import com.jme3.scene.control.CameraControl.ControlDirection; |
|
|
|
import com.jme3.scene.control.CameraControl.ControlDirection; |
|
|
|
import com.jme3.scene.shape.Quad; |
|
|
|
import com.jme3.scene.shape.Quad; |
|
|
|
import com.jme3.system.AppSettings; |
|
|
|
import com.jme3.system.AppSettings; |
|
|
|
import java.awt.event.KeyEvent; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
|
|
|
* A 3rd-person camera node follows a target (teapot). Follow the teapot with |
|
|
|
|
|
|
|
* WASD keys, rotate by dragging the mouse. |
|
|
|
|
|
|
|
*/ |
|
|
|
public class TestCameraNode extends SimpleApplication implements AnalogListener, ActionListener { |
|
|
|
public class TestCameraNode extends SimpleApplication implements AnalogListener, ActionListener { |
|
|
|
|
|
|
|
|
|
|
|
private Geometry teaGeom; |
|
|
|
private Geometry teaGeom; |
|
|
|
private Node teaNode; |
|
|
|
private Node teaNode; |
|
|
|
CameraNode camNode; |
|
|
|
CameraNode camNode; |
|
|
|
|
|
|
|
boolean rotate = false; |
|
|
|
public static void main(String[] args) { |
|
|
|
Vector3f direction = new Vector3f(); |
|
|
|
TestCameraNode app = new TestCameraNode(); |
|
|
|
|
|
|
|
AppSettings s = new AppSettings(true); |
|
|
|
public static void main(String[] args) { |
|
|
|
s.setFrameRate(100); |
|
|
|
TestCameraNode app = new TestCameraNode(); |
|
|
|
app.setSettings(s); |
|
|
|
AppSettings s = new AppSettings(true); |
|
|
|
app.start(); |
|
|
|
s.setFrameRate(100); |
|
|
|
|
|
|
|
app.setSettings(s); |
|
|
|
|
|
|
|
app.start(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void simpleInitApp() { |
|
|
|
|
|
|
|
// load a teapot model
|
|
|
|
|
|
|
|
teaGeom = (Geometry) assetManager.loadModel("Models/Teapot/Teapot.obj"); |
|
|
|
|
|
|
|
Material mat = new Material(assetManager, "Common/MatDefs/Misc/ShowNormals.j3md"); |
|
|
|
|
|
|
|
teaGeom.setMaterial(mat); |
|
|
|
|
|
|
|
//create a node to attach the geometry and the camera node
|
|
|
|
|
|
|
|
teaNode = new Node("teaNode"); |
|
|
|
|
|
|
|
teaNode.attachChild(teaGeom); |
|
|
|
|
|
|
|
rootNode.attachChild(teaNode); |
|
|
|
|
|
|
|
// create a floor
|
|
|
|
|
|
|
|
mat = new Material(assetManager, "Common/MatDefs/Misc/Unshaded.j3md"); |
|
|
|
|
|
|
|
mat.setTexture("ColorMap", assetManager.loadTexture("Interface/Logo/Monkey.jpg")); |
|
|
|
|
|
|
|
Geometry ground = new Geometry("ground", new Quad(50, 50)); |
|
|
|
|
|
|
|
ground.setLocalRotation(new Quaternion().fromAngleAxis(-FastMath.HALF_PI, Vector3f.UNIT_X)); |
|
|
|
|
|
|
|
ground.setLocalTranslation(-25, -1, 25); |
|
|
|
|
|
|
|
ground.setMaterial(mat); |
|
|
|
|
|
|
|
rootNode.attachChild(ground); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//creating the camera Node
|
|
|
|
|
|
|
|
camNode = new CameraNode("CamNode", cam); |
|
|
|
|
|
|
|
//Setting the direction to Spatial to camera, this means the camera will copy the movements of the Node
|
|
|
|
|
|
|
|
camNode.setControlDir(ControlDirection.SpatialToCamera); |
|
|
|
|
|
|
|
//attaching the camNode to the teaNode
|
|
|
|
|
|
|
|
teaNode.attachChild(camNode); |
|
|
|
|
|
|
|
//setting the local translation of the cam node to move it away from the teanNode a bit
|
|
|
|
|
|
|
|
camNode.setLocalTranslation(new Vector3f(-10, 0, 0)); |
|
|
|
|
|
|
|
//setting the camNode to look at the teaNode
|
|
|
|
|
|
|
|
camNode.lookAt(teaNode.getLocalTranslation(), Vector3f.UNIT_Y); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//disable the default 1st-person flyCam (don't forget this!!)
|
|
|
|
|
|
|
|
flyCam.setEnabled(false); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
registerInput(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void registerInput() { |
|
|
|
|
|
|
|
inputManager.addMapping("moveForward", new KeyTrigger(keyInput.KEY_UP), new KeyTrigger(keyInput.KEY_W)); |
|
|
|
|
|
|
|
inputManager.addMapping("moveBackward", new KeyTrigger(keyInput.KEY_DOWN), new KeyTrigger(keyInput.KEY_S)); |
|
|
|
|
|
|
|
inputManager.addMapping("moveRight", new KeyTrigger(keyInput.KEY_RIGHT), new KeyTrigger(keyInput.KEY_D)); |
|
|
|
|
|
|
|
inputManager.addMapping("moveLeft", new KeyTrigger(keyInput.KEY_LEFT), new KeyTrigger(keyInput.KEY_A)); |
|
|
|
|
|
|
|
inputManager.addMapping("toggleRotate", new MouseButtonTrigger(MouseInput.BUTTON_LEFT)); |
|
|
|
|
|
|
|
inputManager.addMapping("rotateRight", new MouseAxisTrigger(MouseInput.AXIS_X, true)); |
|
|
|
|
|
|
|
inputManager.addMapping("rotateLeft", new MouseAxisTrigger(MouseInput.AXIS_X, false)); |
|
|
|
|
|
|
|
inputManager.addListener(this, "moveForward", "moveBackward", "moveRight", "moveLeft"); |
|
|
|
|
|
|
|
inputManager.addListener(this, "rotateRight", "rotateLeft", "toggleRotate"); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void onAnalog(String name, float value, float tpf) { |
|
|
|
|
|
|
|
//computing the normalized direction of the cam to move the teaNode
|
|
|
|
|
|
|
|
direction.set(cam.getDirection()).normalizeLocal(); |
|
|
|
|
|
|
|
if (name.equals("moveForward")) { |
|
|
|
|
|
|
|
direction.multLocal(5 * tpf); |
|
|
|
|
|
|
|
teaNode.move(direction); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("moveBackward")) { |
|
|
|
public void simpleInitApp() { |
|
|
|
direction.multLocal(-5 * tpf); |
|
|
|
// Load a teapot model
|
|
|
|
teaNode.move(direction); |
|
|
|
teaGeom = (Geometry) assetManager.loadModel("Models/Teapot/Teapot.obj"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Material mat = new Material(assetManager, "Common/MatDefs/Misc/ShowNormals.j3md"); |
|
|
|
|
|
|
|
teaGeom.setMaterial(mat); |
|
|
|
|
|
|
|
//creating a node to attach the geometry and the camera node
|
|
|
|
|
|
|
|
teaNode = new Node("teaNode"); |
|
|
|
|
|
|
|
teaNode.attachChild(teaGeom); |
|
|
|
|
|
|
|
mat = new Material(assetManager, "Common/MatDefs/Misc/Unshaded.j3md"); |
|
|
|
|
|
|
|
mat.setTexture("ColorMap", assetManager.loadTexture("Interface/Logo/Monkey.jpg")); |
|
|
|
|
|
|
|
Geometry ground = new Geometry("ground", new Quad(50, 50)); |
|
|
|
|
|
|
|
ground.setLocalRotation(new Quaternion().fromAngleAxis(-FastMath.HALF_PI, Vector3f.UNIT_X)); |
|
|
|
|
|
|
|
ground.setLocalTranslation(-25, -1, 25); |
|
|
|
|
|
|
|
ground.setMaterial(mat); |
|
|
|
|
|
|
|
rootNode.attachChild(ground); |
|
|
|
|
|
|
|
rootNode.attachChild(teaNode); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//creating the camera Node
|
|
|
|
|
|
|
|
camNode = new CameraNode("CamNode", cam); |
|
|
|
|
|
|
|
//Setting the direction to Spatial to camera, this means the camera will copy the movements of the Node
|
|
|
|
|
|
|
|
camNode.setControlDir(ControlDirection.SpatialToCamera); |
|
|
|
|
|
|
|
//attaching the camNode to the teaNode
|
|
|
|
|
|
|
|
teaNode.attachChild(camNode); |
|
|
|
|
|
|
|
//setting the local translation of the cam node to move it away from the teanNode a bit
|
|
|
|
|
|
|
|
camNode.setLocalTranslation(new Vector3f(-10, 0, 0)); |
|
|
|
|
|
|
|
//setting the camNode to look at the teaNode
|
|
|
|
|
|
|
|
camNode.lookAt(teaNode.getLocalTranslation(), Vector3f.UNIT_Y); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//disabling the flyCam (don't forget that!!)
|
|
|
|
|
|
|
|
flyCam.setEnabled(false); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
regsiterInput(); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("moveRight")) { |
|
|
|
public void regsiterInput() { |
|
|
|
direction.crossLocal(Vector3f.UNIT_Y).multLocal(5 * tpf); |
|
|
|
inputManager.addMapping("moveForward", new KeyTrigger(KeyInput.KEY_W)); |
|
|
|
teaNode.move(direction); |
|
|
|
inputManager.addMapping("moveBackward", new KeyTrigger(KeyInput.KEY_S)); |
|
|
|
|
|
|
|
inputManager.addMapping("moveRight", new KeyTrigger(KeyInput.KEY_D)); |
|
|
|
|
|
|
|
inputManager.addMapping("moveLeft", new KeyTrigger(KeyInput.KEY_A)); |
|
|
|
|
|
|
|
inputManager.addMapping("toggleRotate", new MouseButtonTrigger(MouseInput.BUTTON_LEFT)); |
|
|
|
|
|
|
|
inputManager.addMapping("rotateRight", new MouseAxisTrigger(MouseInput.AXIS_X, true)); |
|
|
|
|
|
|
|
inputManager.addMapping("rotateLeft", new MouseAxisTrigger(MouseInput.AXIS_X, false)); |
|
|
|
|
|
|
|
inputManager.addListener(this, "moveForward", "moveBackward", "moveRight", "moveLeft"); |
|
|
|
|
|
|
|
inputManager.addListener(this, "rotateRight", "rotateLeft", "toggleRotate"); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
boolean rotate = false; |
|
|
|
if (name.equals("moveLeft")) { |
|
|
|
Vector3f direction = new Vector3f(); |
|
|
|
direction.crossLocal(Vector3f.UNIT_Y).multLocal(-5 * tpf); |
|
|
|
|
|
|
|
teaNode.move(direction); |
|
|
|
public void onAnalog(String name, float value, float tpf) { |
|
|
|
} |
|
|
|
//computing the normalized direction of the cam to move the teaNode
|
|
|
|
if (name.equals("rotateRight") && rotate) { |
|
|
|
direction.set(cam.getDirection()).normalizeLocal(); |
|
|
|
teaNode.rotate(0, 5 * tpf, 0); |
|
|
|
if (name.equals("moveForward")) { |
|
|
|
} |
|
|
|
direction.multLocal(5 * tpf); |
|
|
|
if (name.equals("rotateLeft") && rotate) { |
|
|
|
teaNode.move(direction); |
|
|
|
teaNode.rotate(0, -5 * tpf, 0); |
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("moveBackward")) { |
|
|
|
|
|
|
|
direction.multLocal(-5 * tpf); |
|
|
|
|
|
|
|
teaNode.move(direction); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("moveRight")) { |
|
|
|
|
|
|
|
direction.crossLocal(Vector3f.UNIT_Y).multLocal(5 * tpf); |
|
|
|
|
|
|
|
teaNode.move(direction); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("moveLeft")) { |
|
|
|
|
|
|
|
direction.crossLocal(Vector3f.UNIT_Y).multLocal(-5 * tpf); |
|
|
|
|
|
|
|
teaNode.move(direction); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("rotateRight") && rotate) { |
|
|
|
|
|
|
|
teaNode.rotate(0, 5 * tpf, 0); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("rotateLeft") && rotate) { |
|
|
|
|
|
|
|
teaNode.rotate(0, -5 * tpf, 0); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
public void onAction(String name, boolean keyPressed, float tpf) { |
|
|
|
} |
|
|
|
//toggling rotation on or off
|
|
|
|
|
|
|
|
if (name.equals("toggleRotate") && keyPressed) { |
|
|
|
|
|
|
|
rotate = true; |
|
|
|
|
|
|
|
inputManager.setCursorVisible(false); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("toggleRotate") && !keyPressed) { |
|
|
|
|
|
|
|
rotate = false; |
|
|
|
|
|
|
|
inputManager.setCursorVisible(true); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void onAction(String name, boolean keyPressed, float tpf) { |
|
|
|
|
|
|
|
//toggling rotation on or off
|
|
|
|
|
|
|
|
if (name.equals("toggleRotate") && keyPressed) { |
|
|
|
|
|
|
|
rotate = true; |
|
|
|
|
|
|
|
inputManager.setCursorVisible(false); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
if (name.equals("toggleRotate") && !keyPressed) { |
|
|
|
|
|
|
|
rotate = false; |
|
|
|
|
|
|
|
inputManager.setCursorVisible(true); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|