Bone animation :
- Fixed squashing flickering when changing animation - Fixed scale computation when blending - Ogre loader can now properly load scales from ogre xml files git-svn-id: https://jmonkeyengine.googlecode.com/svn/trunk@6993 75d07b2b-3a1a-0410-a2c5-0572b91ccdca
This commit is contained in:
parent
ba4265b77a
commit
67a7c1cf22
@ -413,6 +413,7 @@ public final class Bone implements Savable {
|
|||||||
|
|
||||||
localPos.set(initialPos).addLocal(translation);
|
localPos.set(initialPos).addLocal(translation);
|
||||||
localRot.set(initialRot).multLocal(rotation);
|
localRot.set(initialRot).multLocal(rotation);
|
||||||
|
|
||||||
if (scale != null) {
|
if (scale != null) {
|
||||||
localScale.set(initialScale).multLocal(scale);
|
localScale.set(initialScale).multLocal(scale);
|
||||||
}
|
}
|
||||||
@ -440,7 +441,7 @@ public final class Bone implements Savable {
|
|||||||
|
|
||||||
//scale
|
//scale
|
||||||
if (scale != null) {
|
if (scale != null) {
|
||||||
tmpV2.set(initialScale).addLocal(translation);
|
tmpV2.set(initialScale).multLocal(scale);
|
||||||
localScale.interpolate(tmpV2, weight);
|
localScale.interpolate(tmpV2, weight);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -158,9 +158,8 @@ public final class BoneTrack implements Savable {
|
|||||||
int i;
|
int i;
|
||||||
for (i = 0; i < lastFrame && times[i] < time; i++) {
|
for (i = 0; i < lastFrame && times[i] < time; i++) {
|
||||||
startFrame = i;
|
startFrame = i;
|
||||||
|
endFrame = i + 1;
|
||||||
}
|
}
|
||||||
//i is now startFrame+1;
|
|
||||||
endFrame = i ;
|
|
||||||
|
|
||||||
float blend = (time - times[startFrame])
|
float blend = (time - times[startFrame])
|
||||||
/ (times[endFrame] - times[startFrame]);
|
/ (times[endFrame] - times[startFrame]);
|
||||||
@ -183,10 +182,10 @@ public final class BoneTrack implements Savable {
|
|||||||
if (weight != 1f) {
|
if (weight != 1f) {
|
||||||
// tempQ.slerp(Quaternion.IDENTITY, 1f - weight);
|
// tempQ.slerp(Quaternion.IDENTITY, 1f - weight);
|
||||||
// tempV.multLocal(weight);
|
// tempV.multLocal(weight);
|
||||||
target.blendAnimTransforms(tempV, tempQ, scales != null?tempS:null, weight);
|
target.blendAnimTransforms(tempV, tempQ, scales != null ? tempS : null, weight);
|
||||||
// target.setAnimTransforms(tempV, tempQ);
|
// target.setAnimTransforms(tempV, tempQ);
|
||||||
} else {
|
} else {
|
||||||
target.setAnimTransforms(tempV, tempQ, scales != null?tempS:null);
|
target.setAnimTransforms(tempV, tempQ, scales != null ? tempS : null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,7 +29,6 @@
|
|||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package com.jme3.scene.plugins.ogre;
|
package com.jme3.scene.plugins.ogre;
|
||||||
|
|
||||||
import com.jme3.animation.Bone;
|
import com.jme3.animation.Bone;
|
||||||
@ -42,13 +41,12 @@ import com.jme3.asset.AssetManager;
|
|||||||
import com.jme3.math.Quaternion;
|
import com.jme3.math.Quaternion;
|
||||||
import com.jme3.math.Vector3f;
|
import com.jme3.math.Vector3f;
|
||||||
import com.jme3.util.xml.SAXUtil;
|
import com.jme3.util.xml.SAXUtil;
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.Iterator;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Stack;
|
import java.util.Stack;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
@ -62,64 +60,57 @@ import org.xml.sax.helpers.XMLReaderFactory;
|
|||||||
public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
||||||
|
|
||||||
private static final Logger logger = Logger.getLogger(SceneLoader.class.getName());
|
private static final Logger logger = Logger.getLogger(SceneLoader.class.getName());
|
||||||
|
|
||||||
private AssetManager assetManager;
|
private AssetManager assetManager;
|
||||||
private Stack<String> elementStack = new Stack<String>();
|
private Stack<String> elementStack = new Stack<String>();
|
||||||
|
|
||||||
private HashMap<Integer, Bone> indexToBone = new HashMap<Integer, Bone>();
|
private HashMap<Integer, Bone> indexToBone = new HashMap<Integer, Bone>();
|
||||||
private HashMap<String, Bone> nameToBone = new HashMap<String, Bone>();
|
private HashMap<String, Bone> nameToBone = new HashMap<String, Bone>();
|
||||||
|
|
||||||
private BoneTrack track;
|
private BoneTrack track;
|
||||||
private ArrayList<BoneTrack> tracks = new ArrayList<BoneTrack>();
|
private ArrayList<BoneTrack> tracks = new ArrayList<BoneTrack>();
|
||||||
|
|
||||||
private BoneAnimation animation;
|
private BoneAnimation animation;
|
||||||
private ArrayList<BoneAnimation> animations;
|
private ArrayList<BoneAnimation> animations;
|
||||||
|
|
||||||
private Bone bone;
|
private Bone bone;
|
||||||
private Skeleton skeleton;
|
private Skeleton skeleton;
|
||||||
|
|
||||||
private ArrayList<Float> times = new ArrayList<Float>();
|
private ArrayList<Float> times = new ArrayList<Float>();
|
||||||
private ArrayList<Vector3f> translations = new ArrayList<Vector3f>();
|
private ArrayList<Vector3f> translations = new ArrayList<Vector3f>();
|
||||||
private ArrayList<Quaternion> rotations = new ArrayList<Quaternion>();
|
private ArrayList<Quaternion> rotations = new ArrayList<Quaternion>();
|
||||||
|
private ArrayList<Vector3f> scales = new ArrayList<Vector3f>();
|
||||||
private float time = -1;
|
private float time = -1;
|
||||||
private Vector3f position;
|
private Vector3f position;
|
||||||
private Quaternion rotation;
|
private Quaternion rotation;
|
||||||
private Vector3f scale;
|
private Vector3f scale;
|
||||||
|
|
||||||
private float angle;
|
private float angle;
|
||||||
private Vector3f axis;
|
private Vector3f axis;
|
||||||
|
|
||||||
public void startElement(String uri, String localName, String qName, Attributes attribs) throws SAXException{
|
public void startElement(String uri, String localName, String qName, Attributes attribs) throws SAXException {
|
||||||
if (qName.equals("position") || qName.equals("translate")){
|
if (qName.equals("position") || qName.equals("translate")) {
|
||||||
position = SAXUtil.parseVector3(attribs);
|
position = SAXUtil.parseVector3(attribs);
|
||||||
}else if (qName.equals("rotation") || qName.equals("rotate")){
|
} else if (qName.equals("rotation") || qName.equals("rotate")) {
|
||||||
angle = SAXUtil.parseFloat(attribs.getValue("angle"));
|
angle = SAXUtil.parseFloat(attribs.getValue("angle"));
|
||||||
}else if (qName.equals("axis")){
|
} else if (qName.equals("axis")) {
|
||||||
assert elementStack.peek().equals("rotation")
|
assert elementStack.peek().equals("rotation")
|
||||||
|| elementStack.peek().equals("rotate");
|
|| elementStack.peek().equals("rotate");
|
||||||
axis = SAXUtil.parseVector3(attribs);
|
axis = SAXUtil.parseVector3(attribs);
|
||||||
}else if (qName.equals("scale")){
|
} else if (qName.equals("scale")) {
|
||||||
scale = SAXUtil.parseVector3(attribs);
|
scale = SAXUtil.parseVector3(attribs);
|
||||||
}else if (qName.equals("keyframe")){
|
} else if (qName.equals("keyframe")) {
|
||||||
assert elementStack.peek().equals("keyframes");
|
assert elementStack.peek().equals("keyframes");
|
||||||
time = SAXUtil.parseFloat(attribs.getValue("time"));
|
time = SAXUtil.parseFloat(attribs.getValue("time"));
|
||||||
}else if (qName.equals("keyframes")){
|
} else if (qName.equals("keyframes")) {
|
||||||
assert elementStack.peek().equals("track");
|
assert elementStack.peek().equals("track");
|
||||||
}else if (qName.equals("track")){
|
} else if (qName.equals("track")) {
|
||||||
assert elementStack.peek().equals("tracks");
|
assert elementStack.peek().equals("tracks");
|
||||||
String boneName = SAXUtil.parseString(attribs.getValue("bone"));
|
String boneName = SAXUtil.parseString(attribs.getValue("bone"));
|
||||||
Bone bone = nameToBone.get(boneName);
|
Bone bone = nameToBone.get(boneName);
|
||||||
int index = skeleton.getBoneIndex(bone);
|
int index = skeleton.getBoneIndex(bone);
|
||||||
track = new BoneTrack(index);
|
track = new BoneTrack(index);
|
||||||
}else if (qName.equals("boneparent")){
|
} else if (qName.equals("boneparent")) {
|
||||||
assert elementStack.peek().equals("bonehierarchy");
|
assert elementStack.peek().equals("bonehierarchy");
|
||||||
String boneName = attribs.getValue("bone");
|
String boneName = attribs.getValue("bone");
|
||||||
String parentName = attribs.getValue("parent");
|
String parentName = attribs.getValue("parent");
|
||||||
Bone bone = nameToBone.get(boneName);
|
Bone bone = nameToBone.get(boneName);
|
||||||
Bone parent = nameToBone.get(parentName);
|
Bone parent = nameToBone.get(parentName);
|
||||||
parent.addChild(bone);
|
parent.addChild(bone);
|
||||||
}else if (qName.equals("bone")){
|
} else if (qName.equals("bone")) {
|
||||||
assert elementStack.peek().equals("bones");
|
assert elementStack.peek().equals("bones");
|
||||||
|
|
||||||
// insert bone into indexed map
|
// insert bone into indexed map
|
||||||
@ -127,65 +118,65 @@ public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
|||||||
int id = SAXUtil.parseInt(attribs.getValue("id"));
|
int id = SAXUtil.parseInt(attribs.getValue("id"));
|
||||||
indexToBone.put(id, bone);
|
indexToBone.put(id, bone);
|
||||||
nameToBone.put(bone.getName(), bone);
|
nameToBone.put(bone.getName(), bone);
|
||||||
}else if (qName.equals("tracks")){
|
} else if (qName.equals("tracks")) {
|
||||||
assert elementStack.peek().equals("animation");
|
assert elementStack.peek().equals("animation");
|
||||||
tracks.clear();
|
tracks.clear();
|
||||||
}else if (qName.equals("animation")){
|
} else if (qName.equals("animation")) {
|
||||||
assert elementStack.peek().equals("animations");
|
assert elementStack.peek().equals("animations");
|
||||||
String name = SAXUtil.parseString(attribs.getValue("name"));
|
String name = SAXUtil.parseString(attribs.getValue("name"));
|
||||||
float length = SAXUtil.parseFloat(attribs.getValue("length"));
|
float length = SAXUtil.parseFloat(attribs.getValue("length"));
|
||||||
animation = new BoneAnimation(name, length);
|
animation = new BoneAnimation(name, length);
|
||||||
}else if (qName.equals("bonehierarchy")){
|
} else if (qName.equals("bonehierarchy")) {
|
||||||
assert elementStack.peek().equals("skeleton");
|
assert elementStack.peek().equals("skeleton");
|
||||||
}else if (qName.equals("animations")){
|
} else if (qName.equals("animations")) {
|
||||||
assert elementStack.peek().equals("skeleton");
|
assert elementStack.peek().equals("skeleton");
|
||||||
animations = new ArrayList<BoneAnimation>();
|
animations = new ArrayList<BoneAnimation>();
|
||||||
}else if (qName.equals("bones")){
|
} else if (qName.equals("bones")) {
|
||||||
assert elementStack.peek().equals("skeleton");
|
assert elementStack.peek().equals("skeleton");
|
||||||
}else if (qName.equals("skeleton")){
|
} else if (qName.equals("skeleton")) {
|
||||||
assert elementStack.size() == 0;
|
assert elementStack.size() == 0;
|
||||||
}
|
}
|
||||||
elementStack.add(qName);
|
elementStack.add(qName);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void endElement(String uri, String name, String qName) {
|
public void endElement(String uri, String name, String qName) {
|
||||||
if (qName.equals("translate") || qName.equals("position")){
|
if (qName.equals("translate") || qName.equals("position") || qName.equals("scale")) {
|
||||||
}else if (qName.equals("axis")){
|
} else if (qName.equals("axis")) {
|
||||||
}else if (qName.equals("rotate") || qName.equals("rotation")){
|
} else if (qName.equals("rotate") || qName.equals("rotation")) {
|
||||||
rotation = new Quaternion();
|
rotation = new Quaternion();
|
||||||
axis.normalizeLocal();
|
axis.normalizeLocal();
|
||||||
rotation.fromAngleNormalAxis(angle, axis);
|
rotation.fromAngleNormalAxis(angle, axis);
|
||||||
angle = 0;
|
angle = 0;
|
||||||
axis = null;
|
axis = null;
|
||||||
}else if (qName.equals("bone")){
|
} else if (qName.equals("bone")) {
|
||||||
bone.setBindTransforms(position, rotation, scale);
|
bone.setBindTransforms(position, rotation, scale);
|
||||||
bone = null;
|
bone = null;
|
||||||
position = null;
|
position = null;
|
||||||
rotation = null;
|
rotation = null;
|
||||||
scale = null;
|
scale = null;
|
||||||
}else if (qName.equals("bonehierarchy")){
|
} else if (qName.equals("bonehierarchy")) {
|
||||||
Bone[] bones = new Bone[indexToBone.size()];
|
Bone[] bones = new Bone[indexToBone.size()];
|
||||||
// find bones without a parent and attach them to the skeleton
|
// find bones without a parent and attach them to the skeleton
|
||||||
// also assign the bones to the bonelist
|
// also assign the bones to the bonelist
|
||||||
for (Map.Entry<Integer, Bone> entry: indexToBone.entrySet()){
|
for (Map.Entry<Integer, Bone> entry : indexToBone.entrySet()) {
|
||||||
Bone bone = entry.getValue();
|
Bone bone = entry.getValue();
|
||||||
bones[entry.getKey()] = bone;
|
bones[entry.getKey()] = bone;
|
||||||
}
|
}
|
||||||
indexToBone.clear();
|
indexToBone.clear();
|
||||||
skeleton = new Skeleton(bones);
|
skeleton = new Skeleton(bones);
|
||||||
}else if (qName.equals("animation")){
|
} else if (qName.equals("animation")) {
|
||||||
animations.add(animation);
|
animations.add(animation);
|
||||||
animation = null;
|
animation = null;
|
||||||
}else if (qName.equals("track")){
|
} else if (qName.equals("track")) {
|
||||||
if (track != null){ // if track has keyframes
|
if (track != null) { // if track has keyframes
|
||||||
tracks.add(track);
|
tracks.add(track);
|
||||||
track = null;
|
track = null;
|
||||||
}
|
}
|
||||||
}else if (qName.equals("tracks")){
|
} else if (qName.equals("tracks")) {
|
||||||
BoneTrack[] trackList = tracks.toArray(new BoneTrack[tracks.size()]);
|
BoneTrack[] trackList = tracks.toArray(new BoneTrack[tracks.size()]);
|
||||||
animation.setTracks(trackList);
|
animation.setTracks(trackList);
|
||||||
tracks.clear();
|
tracks.clear();
|
||||||
}else if (qName.equals("keyframe")){
|
} else if (qName.equals("keyframe")) {
|
||||||
assert time >= 0;
|
assert time >= 0;
|
||||||
assert position != null;
|
assert position != null;
|
||||||
assert rotation != null;
|
assert rotation != null;
|
||||||
@ -193,28 +184,38 @@ public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
|||||||
times.add(time);
|
times.add(time);
|
||||||
translations.add(position);
|
translations.add(position);
|
||||||
rotations.add(rotation);
|
rotations.add(rotation);
|
||||||
|
if (scale != null) {
|
||||||
|
scales.add(scale);
|
||||||
|
}else{
|
||||||
|
scales.add(new Vector3f(1,1,1));
|
||||||
|
}
|
||||||
|
|
||||||
time = -1;
|
time = -1;
|
||||||
position = null;
|
position = null;
|
||||||
rotation = null;
|
rotation = null;
|
||||||
scale = null;
|
scale = null;
|
||||||
}else if (qName.equals("keyframes")){
|
} else if (qName.equals("keyframes")) {
|
||||||
if (times.size() > 0){
|
if (times.size() > 0) {
|
||||||
float[] timesArray = new float[times.size()];
|
float[] timesArray = new float[times.size()];
|
||||||
for (int i = 0; i < timesArray.length; i++)
|
for (int i = 0; i < timesArray.length; i++) {
|
||||||
timesArray[i] = times.get(i);
|
timesArray[i] = times.get(i);
|
||||||
|
}
|
||||||
|
|
||||||
Vector3f[] transArray = translations.toArray(new Vector3f[translations.size()]);
|
Vector3f[] transArray = translations.toArray(new Vector3f[translations.size()]);
|
||||||
Quaternion[] rotArray = rotations.toArray(new Quaternion[rotations.size()]);
|
Quaternion[] rotArray = rotations.toArray(new Quaternion[rotations.size()]);
|
||||||
track.setKeyframes(timesArray, transArray, rotArray);
|
Vector3f[] scalesArray = scales.toArray(new Vector3f[scales.size()]);
|
||||||
}else{
|
|
||||||
|
track.setKeyframes(timesArray, transArray, rotArray, scalesArray);
|
||||||
|
//track.setKeyframes(timesArray, transArray, rotArray);
|
||||||
|
} else {
|
||||||
track = null;
|
track = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
times.clear();
|
times.clear();
|
||||||
translations.clear();
|
translations.clear();
|
||||||
rotations.clear();
|
rotations.clear();
|
||||||
}else if (qName.equals("skeleton")){
|
scales.clear();
|
||||||
|
} else if (qName.equals("skeleton")) {
|
||||||
nameToBone.clear();
|
nameToBone.clear();
|
||||||
}
|
}
|
||||||
assert elementStack.peek().equals(qName);
|
assert elementStack.peek().equals(qName);
|
||||||
@ -225,15 +226,16 @@ public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
|||||||
* Reset the SkeletonLoader in case an error occured while parsing XML.
|
* Reset the SkeletonLoader in case an error occured while parsing XML.
|
||||||
* This allows future use of the loader even after an error.
|
* This allows future use of the loader even after an error.
|
||||||
*/
|
*/
|
||||||
private void fullReset(){
|
private void fullReset() {
|
||||||
elementStack.clear();
|
elementStack.clear();
|
||||||
indexToBone.clear();
|
indexToBone.clear();
|
||||||
nameToBone.clear();
|
nameToBone.clear();
|
||||||
track = null;
|
track = null;
|
||||||
tracks.clear();
|
tracks.clear();
|
||||||
animation = null;
|
animation = null;
|
||||||
if (animations != null)
|
if (animations != null) {
|
||||||
animations.clear();
|
animations.clear();
|
||||||
|
}
|
||||||
|
|
||||||
bone = null;
|
bone = null;
|
||||||
skeleton = null;
|
skeleton = null;
|
||||||
@ -248,21 +250,21 @@ public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
|||||||
axis = null;
|
axis = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Object load(InputStream in) throws IOException{
|
public Object load(InputStream in) throws IOException {
|
||||||
try{
|
try {
|
||||||
XMLReader xr = XMLReaderFactory.createXMLReader();
|
XMLReader xr = XMLReaderFactory.createXMLReader();
|
||||||
xr.setContentHandler(this);
|
xr.setContentHandler(this);
|
||||||
xr.setErrorHandler(this);
|
xr.setErrorHandler(this);
|
||||||
InputStreamReader r = new InputStreamReader(in);
|
InputStreamReader r = new InputStreamReader(in);
|
||||||
xr.parse(new InputSource(r));
|
xr.parse(new InputSource(r));
|
||||||
if (animations == null){
|
if (animations == null) {
|
||||||
animations = new ArrayList<BoneAnimation>();
|
animations = new ArrayList<BoneAnimation>();
|
||||||
}
|
}
|
||||||
AnimData data = new AnimData(skeleton, animations);
|
AnimData data = new AnimData(skeleton, animations);
|
||||||
skeleton = null;
|
skeleton = null;
|
||||||
animations = null;
|
animations = null;
|
||||||
return data;
|
return data;
|
||||||
} catch (SAXException ex){
|
} catch (SAXException ex) {
|
||||||
IOException ioEx = new IOException("Error while parsing Ogre3D dotScene");
|
IOException ioEx = new IOException("Error while parsing Ogre3D dotScene");
|
||||||
ioEx.initCause(ex);
|
ioEx.initCause(ex);
|
||||||
fullReset();
|
fullReset();
|
||||||
@ -277,5 +279,4 @@ public class SkeletonLoader extends DefaultHandler implements AssetLoader {
|
|||||||
in.close();
|
in.close();
|
||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user