parent
bbb3cf59b3
commit
abe094e74a
@ -0,0 +1,114 @@ |
||||
/* |
||||
* Copyright (c) 2009-2012 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 com.jme3.anim; |
||||
|
||||
import com.jme3.export.*; |
||||
import com.jme3.math.*; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.util.clone.Cloner; |
||||
import com.jme3.util.clone.JmeCloneable; |
||||
|
||||
import java.io.IOException; |
||||
|
||||
/** |
||||
* Contains a list of transforms and times for each keyframe. |
||||
* |
||||
* @author Rémy Bouquet |
||||
*/ |
||||
public final class SpatialTrack extends TransformTrack implements JmeCloneable, Savable { |
||||
|
||||
private Spatial target; |
||||
|
||||
/** |
||||
* Serialization-only. Do not use. |
||||
*/ |
||||
public SpatialTrack() { |
||||
super(); |
||||
} |
||||
|
||||
/** |
||||
* Creates a spatial track for the given Spatial |
||||
* |
||||
* @param target The Spatial target of this track |
||||
* @param times a float array with the time of each frame |
||||
* @param translations the translation of the bone for each frame |
||||
* @param rotations the rotation of the bone for each frame |
||||
* @param scales the scale of the bone for each frame |
||||
*/ |
||||
public SpatialTrack(Spatial target, float[] times, Vector3f[] translations, Quaternion[] rotations, Vector3f[] scales) { |
||||
super(times, translations, rotations, scales); |
||||
this.target = target; |
||||
} |
||||
|
||||
@Override |
||||
public boolean interpolate(double t) { |
||||
setDefaultTransform(target.getLocalTransform()); |
||||
boolean running = super.interpolate(t); |
||||
Transform transform = getInterpolatedTransform(); |
||||
target.setLocalTransform(transform); |
||||
return running; |
||||
} |
||||
|
||||
public void setTarget(Spatial target) { |
||||
this.target = target; |
||||
} |
||||
|
||||
@Override |
||||
public Object jmeClone() { |
||||
try { |
||||
return super.clone(); |
||||
} catch (CloneNotSupportedException e) { |
||||
throw new RuntimeException("Error cloning", e); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void cloneFields(Cloner cloner, Object original) { |
||||
super.cloneFields(cloner, original); |
||||
this.target = cloner.clone(target); |
||||
} |
||||
|
||||
@Override |
||||
public void write(JmeExporter ex) throws IOException { |
||||
super.write(ex); |
||||
OutputCapsule oc = ex.getCapsule(this); |
||||
oc.write(target, "target", null); |
||||
} |
||||
|
||||
@Override |
||||
public void read(JmeImporter im) throws IOException { |
||||
super.read(im); |
||||
InputCapsule ic = im.getCapsule(this); |
||||
target = (Spatial) ic.readSavable("target", null); |
||||
} |
||||
|
||||
} |
@ -1,320 +0,0 @@ |
||||
/* |
||||
* Copyright (c) 2009-2012 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.model; |
||||
|
||||
import com.jme3.animation.*; |
||||
import com.jme3.app.ChaseCameraAppState; |
||||
import com.jme3.app.SimpleApplication; |
||||
import com.jme3.input.KeyInput; |
||||
import com.jme3.input.controls.ActionListener; |
||||
import com.jme3.input.controls.KeyTrigger; |
||||
import com.jme3.math.*; |
||||
import com.jme3.renderer.Limits; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.scene.control.Control; |
||||
import com.jme3.scene.plugins.gltf.GltfModelKey; |
||||
|
||||
import java.util.*; |
||||
|
||||
//import com.jme3.scene.debug.custom.SkeletonDebugAppState;
|
||||
|
||||
public class TestGltfLoading2 extends SimpleApplication { |
||||
|
||||
Node autoRotate = new Node("autoRotate"); |
||||
List<Spatial> assets = new ArrayList<>(); |
||||
Node probeNode; |
||||
float time = 0; |
||||
int assetIndex = 0; |
||||
boolean useAutoRotate = false; |
||||
private final static String indentString = "\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t"; |
||||
int duration = 2; |
||||
boolean playAnim = true; |
||||
|
||||
public static void main(String[] args) { |
||||
TestGltfLoading2 app = new TestGltfLoading2(); |
||||
app.start(); |
||||
} |
||||
|
||||
/* |
||||
WARNING this test case can't wok without the assets, and considering their size, they are not pushed into the repo |
||||
you can find them here : |
||||
https://github.com/KhronosGroup/glTF-Sample-Models/tree/master/2.0
|
||||
https://sketchfab.com/features/gltf
|
||||
You have to copy them in Model/gltf folder in the test-data project. |
||||
*/ |
||||
public void simpleInitApp() { |
||||
|
||||
// SkeletonDebugAppState skeletonDebugAppState = new SkeletonDebugAppState();
|
||||
// getStateManager().attach(skeletonDebugAppState);
|
||||
|
||||
// cam.setLocation(new Vector3f(4.0339394f, 2.645184f, 6.4627485f));
|
||||
// cam.setRotation(new Quaternion(-0.013950467f, 0.98604023f, -0.119502485f, -0.11510504f));
|
||||
cam.setFrustumPerspective(45f, (float) cam.getWidth() / cam.getHeight(), 0.1f, 100f); |
||||
renderer.setDefaultAnisotropicFilter(Math.min(renderer.getLimits().get(Limits.TextureAnisotropy), 8)); |
||||
setPauseOnLostFocus(false); |
||||
|
||||
flyCam.setMoveSpeed(5); |
||||
flyCam.setDragToRotate(true); |
||||
flyCam.setEnabled(false); |
||||
viewPort.setBackgroundColor(new ColorRGBA().setAsSrgb(0.2f, 0.2f, 0.2f, 1.0f)); |
||||
rootNode.attachChild(autoRotate); |
||||
probeNode = (Node) assetManager.loadModel("Scenes/defaultProbe.j3o"); |
||||
autoRotate.attachChild(probeNode); |
||||
|
||||
// DirectionalLight dl = new DirectionalLight();
|
||||
// dl.setDirection(new Vector3f(-1f, -1.0f, -1f).normalizeLocal());
|
||||
// dl.setColor(new ColorRGBA(1f, 1f, 1f, 1.0f));
|
||||
// rootNode.addLight(dl);
|
||||
|
||||
// DirectionalLight dl2 = new DirectionalLight();
|
||||
// dl2.setDirection(new Vector3f(1f, 1.0f, 1f).normalizeLocal());
|
||||
// dl2.setColor(new ColorRGBA(0.5f, 0.5f, 0.5f, 1.0f));
|
||||
// rootNode.addLight(dl2);
|
||||
|
||||
// PointLight pl = new PointLight(new Vector3f(5.0f, 5.0f, 5.0f), ColorRGBA.White, 30);
|
||||
// rootNode.addLight(pl);
|
||||
// PointLight pl1 = new PointLight(new Vector3f(-5.0f, -5.0f, -5.0f), ColorRGBA.White.mult(0.5f), 50);
|
||||
// rootNode.addLight(pl1);
|
||||
//loadModel("Models/gltf/buffalo/scene.gltf", new Vector3f(0, -1, 0), 0.1f);
|
||||
//loadModel("Models/gltf/war/scene.gltf", new Vector3f(0, -1, 0), 0.1f);
|
||||
loadModel("Models/gltf/ganjaarl/scene.gltf", new Vector3f(0, -1, 0), 0.01f); |
||||
//loadModel("Models/gltf/hero/scene.gltf", new Vector3f(0, -1, 0), 0.1f);
|
||||
//loadModel("Models/gltf/mercy/scene.gltf", new Vector3f(0, -1, 0), 0.01f);
|
||||
//loadModel("Models/gltf/crab/scene.gltf", Vector3f.ZERO, 1);
|
||||
//loadModel("Models/gltf/manta/scene.gltf", Vector3f.ZERO, 0.2f);
|
||||
// loadModel("Models/gltf/bone/scene.gltf", Vector3f.ZERO, 0.1f);
|
||||
// loadModel("Models/gltf/box/box.gltf", Vector3f.ZERO, 1);
|
||||
// loadModel("Models/gltf/duck/Duck.gltf", new Vector3f(0, -1, 0), 1);
|
||||
// loadModel("Models/gltf/damagedHelmet/damagedHelmet.gltf", Vector3f.ZERO, 1);
|
||||
// loadModel("Models/gltf/hornet/scene.gltf", new Vector3f(0, -0.5f, 0), 0.4f);
|
||||
//// loadModel("Models/gltf/adamHead/adamHead.gltf", Vector3f.ZERO, 0.6f);
|
||||
//loadModel("Models/gltf/busterDrone/busterDrone.gltf", new Vector3f(0, 0f, 0), 0.8f);
|
||||
// loadModel("Models/gltf/animatedCube/AnimatedCube.gltf", Vector3f.ZERO, 0.5f);
|
||||
//
|
||||
// //loadModel("Models/gltf/BoxAnimated/BoxAnimated.gltf", new Vector3f(0, 0f, 0), 0.8f);
|
||||
//
|
||||
//loadModel("Models/gltf/RiggedFigure/RiggedSimple.gltf", new Vector3f(0, -0.3f, 0), 0.2f);
|
||||
//loadModel("Models/gltf/RiggedFigure/RiggedFigure.gltf", new Vector3f(0, -1f, 0), 1f);
|
||||
//loadModel("Models/gltf/CesiumMan/CesiumMan.gltf", new Vector3f(0, -1, 0), 1f);
|
||||
//loadModel("Models/gltf/BrainStem/BrainStem.gltf", new Vector3f(0, -1, 0), 1f);
|
||||
//loadModel("Models/gltf/Jaime/Jaime.gltf", new Vector3f(0, -1, 0), 1f);
|
||||
//loadModel("Models/gltf/GiantWorm/GiantWorm.gltf", new Vector3f(0, -1, 0), 1f);
|
||||
//loadModel("Models/gltf/RiggedFigure/WalkingLady.gltf", new Vector3f(0, -0.f, 0), 1f);
|
||||
//loadModel("Models/gltf/Monster/Monster.gltf", Vector3f.ZERO, 0.03f);
|
||||
|
||||
// loadModel("Models/gltf/corset/Corset.gltf", new Vector3f(0, -1, 0), 20f);
|
||||
// loadModel("Models/gltf/boxInter/BoxInterleaved.gltf", new Vector3f(0, 0, 0), 1f);
|
||||
|
||||
|
||||
probeNode.attachChild(assets.get(0)); |
||||
|
||||
ChaseCameraAppState chaseCam = new ChaseCameraAppState(); |
||||
chaseCam.setTarget(probeNode); |
||||
getStateManager().attach(chaseCam); |
||||
chaseCam.setInvertHorizontalAxis(true); |
||||
chaseCam.setInvertVerticalAxis(true); |
||||
chaseCam.setZoomSpeed(0.5f); |
||||
chaseCam.setMinVerticalRotation(-FastMath.HALF_PI); |
||||
chaseCam.setRotationSpeed(3); |
||||
chaseCam.setDefaultDistance(3); |
||||
chaseCam.setDefaultVerticalRotation(0.3f); |
||||
|
||||
inputManager.addMapping("autorotate", new KeyTrigger(KeyInput.KEY_SPACE)); |
||||
inputManager.addListener(new ActionListener() { |
||||
@Override |
||||
public void onAction(String name, boolean isPressed, float tpf) { |
||||
if (isPressed) { |
||||
useAutoRotate = !useAutoRotate; |
||||
} |
||||
} |
||||
}, "autorotate"); |
||||
|
||||
inputManager.addMapping("toggleAnim", new KeyTrigger(KeyInput.KEY_RETURN)); |
||||
|
||||
inputManager.addListener(new ActionListener() { |
||||
@Override |
||||
public void onAction(String name, boolean isPressed, float tpf) { |
||||
if (isPressed) { |
||||
playAnim = !playAnim; |
||||
if (playAnim) { |
||||
playFirstAnim(rootNode); |
||||
} else { |
||||
stopAnim(rootNode); |
||||
} |
||||
} |
||||
} |
||||
}, "toggleAnim"); |
||||
inputManager.addMapping("nextAnim", new KeyTrigger(KeyInput.KEY_RIGHT)); |
||||
inputManager.addListener(new ActionListener() { |
||||
@Override |
||||
public void onAction(String name, boolean isPressed, float tpf) { |
||||
if (isPressed && animControl != null) { |
||||
AnimChannel c = animControl.getChannel(0); |
||||
if (c == null) { |
||||
c = animControl.createChannel(); |
||||
} |
||||
String anim = anims.poll(); |
||||
anims.add(anim); |
||||
c.setAnim(anim); |
||||
} |
||||
} |
||||
}, "nextAnim"); |
||||
|
||||
dumpScene(rootNode, 0); |
||||
} |
||||
|
||||
private <T extends Control> T findControl(Spatial s, Class<T> controlClass) { |
||||
T ctrl = s.getControl(controlClass); |
||||
if (ctrl != null) { |
||||
return ctrl; |
||||
} |
||||
if (s instanceof Node) { |
||||
Node n = (Node) s; |
||||
for (Spatial spatial : n.getChildren()) { |
||||
ctrl = findControl(spatial, controlClass); |
||||
if (ctrl != null) { |
||||
return ctrl; |
||||
} |
||||
} |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
private void loadModel(String path, Vector3f offset, float scale) { |
||||
GltfModelKey k = new GltfModelKey(path); |
||||
//k.setKeepSkeletonPose(true);
|
||||
Spatial s = assetManager.loadModel(k); |
||||
s.scale(scale); |
||||
s.move(offset); |
||||
assets.add(s); |
||||
if (playAnim) { |
||||
playFirstAnim(s); |
||||
} |
||||
|
||||
SkeletonControl ctrl = findControl(s, SkeletonControl.class); |
||||
|
||||
// ctrl.getSpatial().removeControl(ctrl);
|
||||
if (ctrl == null) { |
||||
return; |
||||
} |
||||
//System.err.println(ctrl.getArmature().toString());
|
||||
//ctrl.setHardwareSkinningPreferred(false);
|
||||
// getStateManager().getState(SkeletonDebugAppState.class).addSkeleton(ctrl, true);
|
||||
// AnimControl aCtrl = findControl(s, AnimControl.class);
|
||||
// //ctrl.getSpatial().removeControl(ctrl);
|
||||
// if (aCtrl == null) {
|
||||
// return;
|
||||
// }
|
||||
// if (aCtrl.getArmature() != null) {
|
||||
// getStateManager().getState(SkeletonDebugAppState.class).addSkeleton(aCtrl.getArmature(), aCtrl.getSpatial(), true);
|
||||
// }
|
||||
|
||||
} |
||||
|
||||
Queue<String> anims = new LinkedList<>(); |
||||
AnimControl animControl; |
||||
|
||||
private void playFirstAnim(Spatial s) { |
||||
|
||||
AnimControl control = s.getControl(AnimControl.class); |
||||
if (control != null) { |
||||
anims.clear(); |
||||
for (String name : control.getAnimationNames()) { |
||||
anims.add(name); |
||||
} |
||||
if (anims.isEmpty()) { |
||||
return; |
||||
} |
||||
String anim = anims.poll(); |
||||
anims.add(anim); |
||||
control.createChannel().setAnim(anim); |
||||
animControl = control; |
||||
} |
||||
if (s instanceof Node) { |
||||
Node n = (Node) s; |
||||
for (Spatial spatial : n.getChildren()) { |
||||
playFirstAnim(spatial); |
||||
} |
||||
} |
||||
} |
||||
|
||||
private void stopAnim(Spatial s) { |
||||
|
||||
AnimControl control = s.getControl(AnimControl.class); |
||||
if (control != null) { |
||||
for (int i = 0; i < control.getNumChannels(); i++) { |
||||
AnimChannel ch = control.getChannel(i); |
||||
ch.reset(true); |
||||
} |
||||
control.clearChannels(); |
||||
} |
||||
if (s instanceof Node) { |
||||
Node n = (Node) s; |
||||
for (Spatial spatial : n.getChildren()) { |
||||
stopAnim(spatial); |
||||
} |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void simpleUpdate(float tpf) { |
||||
|
||||
if (!useAutoRotate) { |
||||
return; |
||||
} |
||||
time += tpf; |
||||
autoRotate.rotate(0, tpf * 0.5f, 0); |
||||
if (time > duration) { |
||||
assets.get(assetIndex).removeFromParent(); |
||||
assetIndex = (assetIndex + 1) % assets.size(); |
||||
if (assetIndex == 0) { |
||||
duration = 10; |
||||
} |
||||
probeNode.attachChild(assets.get(assetIndex)); |
||||
time = 0; |
||||
} |
||||
} |
||||
|
||||
private void dumpScene(Spatial s, int indent) { |
||||
System.err.println(indentString.substring(0, indent) + s.getName() + " (" + s.getClass().getSimpleName() + ") / " + |
||||
s.getLocalTransform().getTranslation().toString() + ", " + |
||||
s.getLocalTransform().getRotation().toString() + ", " + |
||||
s.getLocalTransform().getScale().toString()); |
||||
if (s instanceof Node) { |
||||
Node n = (Node) s; |
||||
for (Spatial spatial : n.getChildren()) { |
||||
dumpScene(spatial, indent + 1); |
||||
} |
||||
} |
||||
} |
||||
} |
Loading…
Reference in new issue