diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/RotationOrder.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/RotationOrder.java
new file mode 100644
index 000000000..8a1e05b36
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/RotationOrder.java
@@ -0,0 +1,52 @@
+package com.jme3.scene.plugins.fbx;
+
+import com.jme3.math.FastMath;
+import com.jme3.math.Quaternion;
+import com.jme3.math.Vector3f;
+
+public enum RotationOrder {
+
+ EULER_XYZ, EULER_XZY, EULER_YZX, EULER_YXZ, EULER_ZXY, EULER_ZYX, SPHERIC_XYZ;
+
+ // Static values field for fast access by an oridinal without Enum.values() overhead
+ public static final RotationOrder[] values = values();
+
+ private RotationOrder() {
+ }
+
+ public Quaternion rotate(Vector3f vec) {
+ return fromEuler(vec.x * FastMath.DEG_TO_RAD, vec.y * FastMath.DEG_TO_RAD, vec.z * FastMath.DEG_TO_RAD, this);
+ }
+
+ public Quaternion rotate(float x, float y, float z) {
+ return fromEuler(x * FastMath.DEG_TO_RAD, y * FastMath.DEG_TO_RAD, z * FastMath.DEG_TO_RAD, this);
+ }
+
+ private static Quaternion fromEuler(float x, float y, float z, RotationOrder order) {
+ switch(order) {
+ case EULER_XYZ:
+ return toQuat(x, Vector3f.UNIT_X, y, Vector3f.UNIT_Y, z, Vector3f.UNIT_Z);
+ case EULER_YXZ:
+ return toQuat(y, Vector3f.UNIT_Y, x, Vector3f.UNIT_X, z, Vector3f.UNIT_Z);
+ case EULER_ZXY:
+ return toQuat(z, Vector3f.UNIT_Z, x, Vector3f.UNIT_X, y, Vector3f.UNIT_Y);
+ case EULER_ZYX:
+ return toQuat(z, Vector3f.UNIT_Z, y, Vector3f.UNIT_Y, x, Vector3f.UNIT_X);
+ case EULER_YZX:
+ return toQuat(y, Vector3f.UNIT_Y, z, Vector3f.UNIT_Z, x, Vector3f.UNIT_X);
+ case EULER_XZY:
+ return toQuat(x, Vector3f.UNIT_X, z, Vector3f.UNIT_Z, y, Vector3f.UNIT_Y);
+ case SPHERIC_XYZ:
+ default:
+ throw new IllegalArgumentException("Spheric rotation is unsupported in this importer");
+ }
+ }
+
+ private static Quaternion toQuat(float ax1v, Vector3f ax1, float ax2v, Vector3f ax2, float ax3v, Vector3f ax3) {
+ // TODO It has some potential in optimization
+ Quaternion q1 = new Quaternion().fromAngleNormalAxis(ax1v, ax1);
+ Quaternion q2 = new Quaternion().fromAngleNormalAxis(ax2v, ax2);
+ Quaternion q3 = new Quaternion().fromAngleNormalAxis(ax3v, ax3);
+ return q1.multLocal(q2).multLocal(q3);
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/SceneLoader.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/SceneLoader.java
index 87e13abed..accbdbbf4 100644
--- a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/SceneLoader.java
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/SceneLoader.java
@@ -1,46 +1,13 @@
-/*
- * Copyright (c) 2009-2014 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.scene.plugins.fbx;
-import java.io.File;
import java.io.IOException;
import java.io.InputStream;
-import java.nio.ByteBuffer;
-import java.nio.FloatBuffer;
import java.util.ArrayList;
-import java.util.LinkedList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
+import java.util.Set;
+import java.util.TreeSet;
import java.util.logging.Level;
import java.util.logging.Logger;
@@ -50,94 +17,83 @@ import com.jme3.animation.Bone;
import com.jme3.animation.BoneTrack;
import com.jme3.animation.Skeleton;
import com.jme3.animation.SkeletonControl;
+import com.jme3.animation.Track;
import com.jme3.asset.AssetInfo;
import com.jme3.asset.AssetKey;
-import com.jme3.asset.AssetLoadException;
import com.jme3.asset.AssetLoader;
import com.jme3.asset.AssetManager;
-import com.jme3.asset.ModelKey;
-import com.jme3.material.Material;
-import com.jme3.material.RenderState.BlendMode;
-import com.jme3.math.ColorRGBA;
-import com.jme3.math.FastMath;
-import com.jme3.math.Matrix4f;
import com.jme3.math.Quaternion;
-import com.jme3.math.Transform;
import com.jme3.math.Vector3f;
-import com.jme3.scene.Geometry;
-import com.jme3.scene.Mesh;
import com.jme3.scene.Node;
-import com.jme3.scene.Spatial;
-import com.jme3.scene.VertexBuffer;
-import com.jme3.scene.Mesh.Mode;
-import com.jme3.scene.VertexBuffer.Type;
-import com.jme3.scene.VertexBuffer.Usage;
import com.jme3.scene.plugins.fbx.AnimationList.AnimInverval;
import com.jme3.scene.plugins.fbx.file.FbxElement;
import com.jme3.scene.plugins.fbx.file.FbxFile;
import com.jme3.scene.plugins.fbx.file.FbxReader;
-import com.jme3.texture.Image;
-import com.jme3.texture.Texture;
-import com.jme3.texture.Texture2D;
-import com.jme3.util.BufferUtils;
+import com.jme3.scene.plugins.fbx.objects.FbxAnimCurve;
+import com.jme3.scene.plugins.fbx.objects.FbxAnimNode;
+import com.jme3.scene.plugins.fbx.objects.FbxBindPose;
+import com.jme3.scene.plugins.fbx.objects.FbxCluster;
+import com.jme3.scene.plugins.fbx.objects.FbxImage;
+import com.jme3.scene.plugins.fbx.objects.FbxMaterial;
+import com.jme3.scene.plugins.fbx.objects.FbxMesh;
+import com.jme3.scene.plugins.fbx.objects.FbxNode;
+import com.jme3.scene.plugins.fbx.objects.FbxObject;
+import com.jme3.scene.plugins.fbx.objects.FbxSkin;
+import com.jme3.scene.plugins.fbx.objects.FbxTexture;
/**
* FBX file format loader
*
Loads scene meshes, materials, textures, skeleton and skeletal animation.
- * Multiple animations can be defined with {@link AnimationList} passing into {@link SceneKey}.
- *
- * @author Aleksandra Menshchikova
+ * Multiple animations can be defined with {@link AnimationList} passing into {@link SceneKey}
+ * or loaded from different animation layer.
*/
public class SceneLoader implements AssetLoader {
private static final Logger logger = Logger.getLogger(SceneLoader.class.getName());
+ private static final double secondsPerUnit = 1 / 46186158000d; // Animation speed factor
+ public static final boolean WARN_IGNORED_ATTRIBUTES = false;
- private AssetManager assetManager;
+ // Scene loading data
+ private List warnings = new ArrayList<>();
private AnimationList animList;
-
private String sceneName;
- private String sceneFilename;
- private String sceneFolderName;
- private float unitSize;
- private float animFrameRate;
- private final double secondsPerUnit = 1 / 46186158000d; // Animation speed factor
+ public String sceneFilename;
+ public String sceneFolderName;
+ public AssetManager assetManager;
+ public AssetInfo currentAssetInfo;
- // Loaded objects data
- private Map meshDataMap = new HashMap();
- private Map matDataMap = new HashMap();
- private Map texDataMap = new HashMap();
- private Map imgDataMap = new HashMap(); // Video clips
- private Map modelDataMap = new HashMap(); // Mesh nodes and limb nodes
- private Map poseDataMap = new HashMap(); // Node bind poses
- private Map skinMap = new HashMap(); // Skin for bone clusters
- private Map clusterMap = new HashMap(); // Bone skinning cluster
- private Map acurveMap = new HashMap(); // Animation curves
- private Map anodeMap = new HashMap(); // Animation nodes
- private Map alayerMap = new HashMap(); // Amination layers
- private Map> refMap = new HashMap>(); // Object links
- private Map> propMap = new HashMap>(); // Property links
+ // Scene global settings
+ private float animFrameRate;
+ public float unitSize;
+ public int xAxis = 1;
+ public int yAxis = 1;
+ public int zAxis = 1;
// Scene objects
- private Map modelMap = new HashMap(); // Mesh nodes
- private Map limbMap = new HashMap(); // Bones
- private Map bindMap = new HashMap(); // Node bind poses
- private Map geomMap = new HashMap(); // Mesh geometries
- private Map matMap = new HashMap();
- private Map texMap = new HashMap();
- private Map imgMap = new HashMap();
+ private Map allObjects = new HashMap<>(); // All supported FBX objects
+ private Map skinMap = new HashMap<>(); // Skin for bone clusters
+ private Map alayerMap = new HashMap<>(); // Amination layers
+ public Map modelMap = new HashMap<>(); // Nodes
+ private Map limbMap = new HashMap<>(); // Nodes that are actually bones
+ private Map bindMap = new HashMap<>(); // Node bind poses
+ private Map geomMap = new HashMap<>(); // Mesh geometries
private Skeleton skeleton;
private AnimControl animControl;
+ public Node sceneNode;
+
+ public void warning(String warning) {
+ warnings.add(warning);
+ }
@Override
public Object load(AssetInfo assetInfo) throws IOException {
+ this.currentAssetInfo = assetInfo;
this.assetManager = assetInfo.getManager();
AssetKey> assetKey = assetInfo.getKey();
if(assetKey instanceof SceneKey)
animList = ((SceneKey) assetKey).getAnimations();
- else if(!(assetKey instanceof ModelKey))
- throw new AssetLoadException("Invalid asset key");
InputStream stream = assetInfo.openStream();
- Node sceneNode = null;
+ final Node sceneNode = this.sceneNode = new Node(sceneName + "-scene");
try {
sceneFilename = assetKey.getName();
sceneFolderName = assetKey.getFolder();
@@ -145,413 +101,131 @@ public class SceneLoader implements AssetLoader {
sceneName = sceneFilename.substring(0, sceneFilename.length() - ext.length() - 1);
if(sceneFolderName != null && sceneFolderName.length() > 0)
sceneName = sceneName.substring(sceneFolderName.length());
- reset();
loadScene(stream);
- sceneNode = linkScene();
+ linkScene();
+ if(warnings.size() > 0)
+ logger.log(Level.WARNING, "Model load finished with warnings:\n" + join(warnings, "\n"));
} finally {
releaseObjects();
- if(stream != null) {
+ if(stream != null)
stream.close();
- }
}
return sceneNode;
}
- private void reset() {
- unitSize = 1;
- animFrameRate = 30;
- }
-
private void loadScene(InputStream stream) throws IOException {
logger.log(Level.FINE, "Loading scene {0}", sceneFilename);
long startTime = System.currentTimeMillis();
FbxFile scene = FbxReader.readFBX(stream);
for(FbxElement e : scene.rootElements) {
- if(e.id.equals("GlobalSettings"))
+ // Is it possible for elements to be in wrong order?
+ switch(e.id) {
+ case "GlobalSettings":
loadGlobalSettings(e);
- else if(e.id.equals("Objects"))
+ break;
+ case "Objects":
loadObjects(e);
- else if(e.id.equals("Connections"))
+ break;
+ case "Connections":
loadConnections(e);
+ break;
+ }
}
long estimatedTime = System.currentTimeMillis() - startTime;
logger.log(Level.FINE, "Loading done in {0} ms", estimatedTime);
}
private void loadGlobalSettings(FbxElement element) {
+ for(FbxElement e2 : element.getFbxProperties()) {
+ String propName = (String) e2.properties.get(0);
+ switch(propName) {
+ case "UnitScaleFactor":
+ this.unitSize = ((Double) e2.properties.get(4)).floatValue();
+ break;
+ case "CustomFrameRate":
+ float framerate = ((Double) e2.properties.get(4)).floatValue();
+ if(framerate != -1)
+ this.animFrameRate = framerate;
+ break;
+ case "UpAxisSign":
+ this.yAxis = ((Integer) e2.properties.get(4)).intValue();
+ break;
+ case "FrontAxisSign":
+ this.zAxis = ((Integer) e2.properties.get(4)).intValue();
+ break;
+ case "CoordAxisSign":
+ this.xAxis = ((Integer) e2.properties.get(4)).intValue();
+ break;
+ }
+ }
+ }
+
+ private void loadObjects(FbxElement element) throws IOException {
+ FbxObject obj = null;
for(FbxElement e : element.children) {
- if(e.id.equals("Properties70")) {
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("P")) {
- String propName = (String) e2.properties.get(0);
- if(propName.equals("UnitScaleFactor"))
- this.unitSize = ((Double) e2.properties.get(4)).floatValue();
- else if(propName.equals("CustomFrameRate")) {
- float framerate = ((Double) e2.properties.get(4)).floatValue();
- if(framerate != -1)
- this.animFrameRate = framerate;
- }
- }
- }
- }
- }
- }
-
- private void loadObjects(FbxElement element) {
- for(FbxElement e : element.children) {
- if(e.id.equals("Geometry"))
- loadGeometry(e);
- else if(e.id.equals("Material"))
- loadMaterial(e);
- else if(e.id.equals("Model"))
- loadModel(e);
- else if(e.id.equals("Pose"))
- loadPose(e);
- else if(e.id.equals("Texture"))
- loadTexture(e);
- else if(e.id.equals("Video"))
- loadImage(e);
- else if(e.id.equals("Deformer"))
- loadDeformer(e);
- else if(e.id.equals("AnimationLayer"))
- loadAnimLayer(e);
- else if(e.id.equals("AnimationCurve"))
- loadAnimCurve(e);
- else if(e.id.equals("AnimationCurveNode"))
- loadAnimNode(e);
- }
- }
-
- private void loadGeometry(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String type = (String) element.properties.get(2);
- if(type.equals("Mesh")) {
- MeshData data = new MeshData();
- for(FbxElement e : element.children) {
- if(e.id.equals("Vertices"))
- data.vertices = (double[]) e.properties.get(0);
- else if(e.id.equals("PolygonVertexIndex"))
- data.indices = (int[]) e.properties.get(0);
- // TODO edges are not used now
- //else if(e.id.equals("Edges"))
- // data.edges = (int[]) e.properties.get(0);
- else if(e.id.equals("LayerElementNormal"))
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("MappingInformationType")) {
- data.normalsMapping = (String) e2.properties.get(0);
- if(!data.normalsMapping.equals("ByVertice") && !data.normalsMapping.equals("ByPolygonVertex"))
- throw new AssetLoadException("Not supported LayerElementNormal.MappingInformationType = " + data.normalsMapping);
- } else if(e2.id.equals("ReferenceInformationType")) {
- data.normalsReference = (String) e2.properties.get(0);
- if(!data.normalsReference.equals("Direct"))
- throw new AssetLoadException("Not supported LayerElementNormal.ReferenceInformationType = " + data.normalsReference);
- } else if(e2.id.equals("Normals"))
- data.normals = (double[]) e2.properties.get(0);
- }
- else if(e.id.equals("LayerElementTangent"))
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("MappingInformationType")) {
- data.tangentsMapping = (String) e2.properties.get(0);
- if(!data.tangentsMapping.equals("ByVertice") && !data.tangentsMapping.equals("ByPolygonVertex"))
- throw new AssetLoadException("Not supported LayerElementTangent.MappingInformationType = " + data.tangentsMapping);
- } else if(e2.id.equals("ReferenceInformationType")) {
- data.tangentsReference = (String) e2.properties.get(0);
- if(!data.tangentsReference.equals("Direct"))
- throw new AssetLoadException("Not supported LayerElementTangent.ReferenceInformationType = " + data.tangentsReference);
- } else if(e2.id.equals("Tangents"))
- data.tangents = (double[]) e2.properties.get(0);
- }
- else if(e.id.equals("LayerElementBinormal"))
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("MappingInformationType")) {
- data.binormalsMapping = (String) e2.properties.get(0);
- if(!data.binormalsMapping.equals("ByVertice") && !data.binormalsMapping.equals("ByPolygonVertex"))
- throw new AssetLoadException("Not supported LayerElementBinormal.MappingInformationType = " + data.binormalsMapping);
- } else if(e2.id.equals("ReferenceInformationType")) {
- data.binormalsReference = (String) e2.properties.get(0);
- if(!data.binormalsReference.equals("Direct"))
- throw new AssetLoadException("Not supported LayerElementBinormal.ReferenceInformationType = " + data.binormalsReference);
- } else if(e2.id.equals("Tangents"))
- data.binormals = (double[]) e2.properties.get(0);
- }
- else if(e.id.equals("LayerElementUV"))
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("MappingInformationType")) {
- data.uvMapping = (String) e2.properties.get(0);
- if(!data.uvMapping.equals("ByPolygonVertex"))
- throw new AssetLoadException("Not supported LayerElementUV.MappingInformationType = " + data.uvMapping);
- } else if(e2.id.equals("ReferenceInformationType")) {
- data.uvReference = (String) e2.properties.get(0);
- if(!data.uvReference.equals("IndexToDirect"))
- throw new AssetLoadException("Not supported LayerElementUV.ReferenceInformationType = " + data.uvReference);
- } else if(e2.id.equals("UV"))
- data.uv = (double[]) e2.properties.get(0);
- else if(e2.id.equals("UVIndex"))
- data.uvIndex = (int[]) e2.properties.get(0);
- }
- // TODO smoothing is not used now
- //else if(e.id.equals("LayerElementSmoothing"))
- // for(FbxElement e2 : e.children) {
- // if(e2.id.equals("MappingInformationType")) {
- // data.smoothingMapping = (String) e2.properties.get(0);
- // if(!data.smoothingMapping.equals("ByEdge"))
- // throw new AssetLoadException("Not supported LayerElementSmoothing.MappingInformationType = " + data.smoothingMapping);
- // } else if(e2.id.equals("ReferenceInformationType")) {
- // data.smoothingReference = (String) e2.properties.get(0);
- // if(!data.smoothingReference.equals("Direct"))
- // throw new AssetLoadException("Not supported LayerElementSmoothing.ReferenceInformationType = " + data.smoothingReference);
- // } else if(e2.id.equals("Smoothing"))
- // data.smoothing = (int[]) e2.properties.get(0);
- // }
- else if(e.id.equals("LayerElementMaterial"))
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("MappingInformationType")) {
- data.materialsMapping = (String) e2.properties.get(0);
- if(!data.materialsMapping.equals("AllSame"))
- throw new AssetLoadException("Not supported LayerElementMaterial.MappingInformationType = " + data.materialsMapping);
- } else if(e2.id.equals("ReferenceInformationType")) {
- data.materialsReference = (String) e2.properties.get(0);
- if(!data.materialsReference.equals("IndexToDirect"))
- throw new AssetLoadException("Not supported LayerElementMaterial.ReferenceInformationType = " + data.materialsReference);
- } else if(e2.id.equals("Materials"))
- data.materials = (int[]) e2.properties.get(0);
- }
- }
- meshDataMap.put(id, data);
- }
- }
-
- private void loadMaterial(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
- String type = (String) element.properties.get(2);
- if(type.equals("")) {
- MaterialData data = new MaterialData();
- data.name = path.substring(0, path.indexOf(0));
- for(FbxElement e : element.children) {
- if(e.id.equals("ShadingModel")) {
- data.shadingModel = (String) e.properties.get(0);
- } else if(e.id.equals("Properties70")) {
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("P")) {
- String propName = (String) e2.properties.get(0);
- if(propName.equals("AmbientColor")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.ambientColor.set((float) x, (float) y, (float) z);
- } else if(propName.equals("AmbientFactor")) {
- double x = (Double) e2.properties.get(4);
- data.ambientFactor = (float) x;
- } else if(propName.equals("DiffuseColor")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.diffuseColor.set((float) x, (float) y, (float) z);
- } else if(propName.equals("DiffuseFactor")) {
- double x = (Double) e2.properties.get(4);
- data.diffuseFactor = (float) x;
- } else if(propName.equals("SpecularColor")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.specularColor.set((float) x, (float) y, (float) z);
- } else if(propName.equals("Shininess") || propName.equals("ShininessExponent")) {
- double x = (Double) e2.properties.get(4);
- data.shininessExponent = (float) x;
- }
- }
- }
- }
- }
- matDataMap.put(id, data);
- }
- }
-
- private void loadModel(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
- String type = (String) element.properties.get(2);
- ModelData data = new ModelData();
- data.name = path.substring(0, path.indexOf(0));
- data.type = type;
- for(FbxElement e : element.children) {
- if(e.id.equals("Properties70")) {
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("P")) {
- String propName = (String) e2.properties.get(0);
- if(propName.equals("Lcl Translation")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.localTranslation.set((float) x, (float) y, (float) z).divideLocal(unitSize);
- } else if(propName.equals("Lcl Rotation")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.localRotation.fromAngles((float) x * FastMath.DEG_TO_RAD, (float) y * FastMath.DEG_TO_RAD, (float) z * FastMath.DEG_TO_RAD);
- } else if(propName.equals("Lcl Scaling")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.localScale.set((float) x, (float) y, (float) z).multLocal(unitSize);
- } else if(propName.equals("PreRotation")) {
- double x = (Double) e2.properties.get(4);
- double y = (Double) e2.properties.get(5);
- double z = (Double) e2.properties.get(6);
- data.preRotation = quatFromBoneAngles((float) x * FastMath.DEG_TO_RAD, (float) y * FastMath.DEG_TO_RAD, (float) z * FastMath.DEG_TO_RAD);
- }
- }
- }
- }
- }
- modelDataMap.put(id, data);
- }
-
- private void loadPose(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
- String type = (String) element.properties.get(2);
- if(type.equals("BindPose")) {
- BindPoseData data = new BindPoseData();
- data.name = path.substring(0, path.indexOf(0));
- for(FbxElement e : element.children) {
- if(e.id.equals("PoseNode")) {
- NodeTransformData item = new NodeTransformData();
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("Node"))
- item.nodeId = (Long) e2.properties.get(0);
- else if(e2.id.equals("Matrix"))
- item.transform = (double[]) e2.properties.get(0);
- }
- data.list.add(item);
- }
- }
- poseDataMap.put(id, data);
- }
- }
-
- private void loadTexture(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
- String type = (String) element.properties.get(2);
- if(type.equals("")) {
- TextureData data = new TextureData();
- data.name = path.substring(0, path.indexOf(0));
- for(FbxElement e : element.children) {
- if(e.id.equals("Type"))
- data.bindType = (String) e.properties.get(0);
- else if(e.id.equals("FileName"))
- data.filename = (String) e.properties.get(0);
- }
- texDataMap.put(id, data);
- }
- }
-
- private void loadImage(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
+ switch(e.id) {
+ case "Geometry":
+ FbxMesh mesh = new FbxMesh(this, e);
+ obj = mesh;
+ if(mesh.geometries != null)
+ geomMap.put(mesh.id, mesh);
+ break;
+ case "Material":
+ obj = new FbxMaterial(this, e);
+ break;
+ case "Model":
+ FbxNode node = new FbxNode(this, e);
+ obj = node;
+ modelMap.put(node.id, node);
+ if(node.isLimb())
+ limbMap.put(node.id, node);
+ break;
+ case "Pose":
+ FbxBindPose pose = new FbxBindPose(this, e);
+ obj = pose;
+ bindMap.put(pose.id, pose);
+ break;
+ case "Texture":
+ obj = new FbxTexture(this, e);
+ break;
+ case "Video":
+ obj = new FbxImage(this, e);
+ break;
+ case "Deformer":
+ obj = loadDeformer(e);
+ break;
+ case "AnimationLayer":
+ FbxObject layer = new FbxObject(this, e);
+ obj = layer;
+ alayerMap.put(layer.id, layer);
+ break;
+ case "AnimationCurve":
+ obj = new FbxAnimCurve(this, e);
+ break;
+ case "AnimationCurveNode":
+ obj = new FbxAnimNode(this, e);
+ break;
+ default:
+ obj = null;
+ //warnings.add("Object with id '" + e.id + "' was ignored");
+ }
+ if(obj != null)
+ allObjects.put(obj.id, obj);
+ }
+ }
+
+ private FbxObject loadDeformer(FbxElement element) {
String type = (String) element.properties.get(2);
- if(type.equals("Clip")) {
- ImageData data = new ImageData();
- data.name = path.substring(0, path.indexOf(0));
- for(FbxElement e : element.children) {
- if(e.id.equals("Type"))
- data.type = (String) e.properties.get(0);
- else if(e.id.equals("FileName"))
- data.filename = (String) e.properties.get(0);
- else if(e.id.equals("RelativeFilename"))
- data.relativeFilename = (String) e.properties.get(0);
- else if(e.id.equals("Content")) {
- if(e.properties.size() > 0)
- data.content = (byte[]) e.properties.get(0);
- }
- }
- imgDataMap.put(id, data);
- }
- }
-
- private void loadDeformer(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String type = (String) element.properties.get(2);
- if(type.equals("Skin")) {
- SkinData skinData = new SkinData();
- for(FbxElement e : element.children) {
- if(e.id.equals("SkinningType"))
- skinData.type = (String) e.properties.get(0);
- }
- skinMap.put(id, skinData);
- } else if(type.equals("Cluster")) {
- ClusterData clusterData = new ClusterData();
- for(FbxElement e : element.children) {
- if(e.id.equals("Indexes"))
- clusterData.indexes = (int[]) e.properties.get(0);
- else if(e.id.equals("Weights"))
- clusterData.weights = (double[]) e.properties.get(0);
- else if(e.id.equals("Transform"))
- clusterData.transform = (double[]) e.properties.get(0);
- else if(e.id.equals("TransformLink"))
- clusterData.transformLink = (double[]) e.properties.get(0);
- }
- clusterMap.put(id, clusterData);
- }
- }
-
- private void loadAnimLayer(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
- String type = (String) element.properties.get(2);
- if(type.equals("")) {
- AnimLayer layer = new AnimLayer();
- layer.name = path.substring(0, path.indexOf(0));
- alayerMap.put(id, layer);
- }
- }
-
- private void loadAnimCurve(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String type = (String) element.properties.get(2);
- if(type.equals("")) {
- AnimCurveData data = new AnimCurveData();
- for(FbxElement e : element.children) {
- if(e.id.equals("KeyTime"))
- data.keyTimes = (long[]) e.properties.get(0);
- else if(e.id.equals("KeyValueFloat"))
- data.keyValues = (float[]) e.properties.get(0);
- }
- acurveMap.put(id, data);
- }
- }
-
- private void loadAnimNode(FbxElement element) {
- long id = (Long) element.properties.get(0);
- String path = (String) element.properties.get(1);
- String type = (String) element.properties.get(2);
- if(type.equals("")) {
- Double x = null, y = null, z = null;
- for(FbxElement e : element.children) {
- if(e.id.equals("Properties70")) {
- for(FbxElement e2 : e.children) {
- if(e2.id.equals("P")) {
- String propName = (String) e2.properties.get(0);
- if(propName.equals("d|X"))
- x = (Double) e2.properties.get(4);
- else if(propName.equals("d|Y"))
- y = (Double) e2.properties.get(4);
- else if(propName.equals("d|Z"))
- z = (Double) e2.properties.get(4);
- }
- }
- }
- }
- // Load only T R S curve nodes
- if(x != null && y != null && z != null) {
- AnimNode node = new AnimNode();
- node.value = new Vector3f(x.floatValue(), y.floatValue(), z.floatValue());
- node.name = path.substring(0, path.indexOf(0));
- anodeMap.put(id, node);
- }
+ switch(type) {
+ case "Skin":
+ FbxSkin skinData = new FbxSkin(this, element);
+ skinMap.put(skinData.id, skinData);
+ return skinData;
+ case "Cluster":
+ FbxCluster clusterData = new FbxCluster(this, element);
+ return clusterData;
}
+ return null;
}
private void loadConnections(FbxElement element) {
@@ -559,698 +233,65 @@ public class SceneLoader implements AssetLoader {
if(e.id.equals("C")) {
String type = (String) e.properties.get(0);
long objId, refId;
- if(type.equals("OO")) {
+ FbxObject obj, ref;
+ switch(type) {
+ case "OO":
objId = (Long) e.properties.get(1);
refId = (Long) e.properties.get(2);
- List links = refMap.get(objId);
- if(links == null) {
- links = new ArrayList();
- refMap.put(objId, links);
+ obj = allObjects.get(objId);
+ ref = allObjects.get(refId);
+ if(ref != null) {
+ ref.link(obj);
+ } else if(refId == 0) {
+ obj.linkToZero();
}
- links.add(refId);
- } else if(type.equals("OP")) {
+ break;
+ case "OP":
objId = (Long) e.properties.get(1);
refId = (Long) e.properties.get(2);
String propName = (String) e.properties.get(3);
- List props = propMap.get(objId);
- if(props == null) {
- props = new ArrayList();
- propMap.put(objId, props);
- }
- props.add(new PropertyLink(refId, propName));
- }
- }
- }
- }
-
- private Geometry createGeomerty(MeshData data) {
- Mesh mesh = new Mesh();
- mesh.setMode(Mode.Triangles);
- // Since each vertex should contain unique texcoord and normal we should unroll vertex indexing
- // So we don't use VertexBuffer.Type.Index for elements drawing
- // Moreover quads should be triangulated (this increases number of vertices)
- boolean isQuads = false;
- if(data.indices != null) {
- data.iCount = data.indices.length;
- data.srcVertexCount = data.vertices.length / 3;
- // Indices contains negative numbers to define polygon last index
- // Check indices strides to be sure we have triangles or quads
- boolean allTriangles = true;
- boolean allQads = true;
- for(int i = 0; i < data.indices.length; ++i) {
- if(i % 3 == 2) { // Triangle stride
- if(data.indices[i] >= 0)
- allTriangles = false;
- } else {
- if(data.indices[i] < 0)
- allTriangles = false;
- }
- if(i % 4 == 3) { // Quad stride
- if(data.indices[i] >= 0)
- allQads = false;
- } else {
- if(data.indices[i] < 0)
- allQads = false;
- }
- }
- if(allTriangles) {
- isQuads = false;
- data.vCount = data.iCount;
- } else if(allQads) {
- isQuads = true;
- data.vCount = 6 * (data.iCount / 4); // Each quad will be splited into two triangles
- } else
- throw new AssetLoadException("Unsupported PolygonVertexIndex stride");
- data.vertexMap = new int[data.vCount];
- data.indexMap = new int[data.vCount];
- // Unroll index array into vertex mapping
- int n = 0;
- for(int i = 0; i < data.iCount; ++i) {
- int index = data.indices[i];
- if(index < 0) {
- int lastIndex = -(index + 1);
- if(isQuads) {
- data.vertexMap[n + 0] = data.indices[i - 3];
- data.vertexMap[n + 1] = data.indices[i - 2];
- data.vertexMap[n + 2] = data.indices[i - 1];
- data.vertexMap[n + 3] = data.indices[i - 3];
- data.vertexMap[n + 4] = data.indices[i - 1];
- data.vertexMap[n + 5] = lastIndex;
- data.indexMap[n + 0] = (i - 3);
- data.indexMap[n + 1] = (i - 2);
- data.indexMap[n + 2] = (i - 1);
- data.indexMap[n + 3] = (i - 3);
- data.indexMap[n + 4] = (i - 1);
- data.indexMap[n + 5] = (i - 0);
- n += 6;
- } else {
- data.vertexMap[n + 0] = data.indices[i - 2];
- data.vertexMap[n + 1] = data.indices[i - 1];
- data.vertexMap[n + 2] = lastIndex;
- data.indexMap[n + 0] = (i - 2);
- data.indexMap[n + 1] = (i - 1);
- data.indexMap[n + 2] = (i - 0);
- n += 3;
- }
- }
- }
- // Build reverse vertex mapping
- data.reverseVertexMap = new ArrayList>(data.srcVertexCount);
- for(int i = 0; i < data.srcVertexCount; ++i)
- data.reverseVertexMap.add(new ArrayList());
- for(int i = 0; i < data.vCount; ++i) {
- int index = data.vertexMap[i];
- data.reverseVertexMap.get(index).add(i);
- }
- } else {
- // Stub for no vertex indexing (direct mapping)
- data.iCount = data.vCount = data.srcVertexCount;
- data.vertexMap = new int[data.vCount];
- data.indexMap = new int[data.vCount];
- data.reverseVertexMap = new ArrayList>(data.vCount);
- for(int i = 0; i < data.vCount; ++i) {
- data.vertexMap[i] = i;
- data.indexMap[i] = i;
- List reverseIndices = new ArrayList(1);
- reverseIndices.add(i);
- data.reverseVertexMap.add(reverseIndices);
- }
- }
- if(data.vertices != null) {
- // Unroll vertices data array
- FloatBuffer posBuf = BufferUtils.createFloatBuffer(data.vCount * 3);
- mesh.setBuffer(VertexBuffer.Type.Position, 3, posBuf);
- int srcCount = data.vertices.length / 3;
- for(int i = 0; i < data.vCount; ++i) {
- int index = data.vertexMap[i];
- if(index > srcCount)
- throw new AssetLoadException("Invalid vertex mapping. Unexpected lookup vertex " + index + " from " + srcCount);
- float x = (float) data.vertices[3 * index + 0] / unitSize;
- float y = (float) data.vertices[3 * index + 1] / unitSize;
- float z = (float) data.vertices[3 * index + 2] / unitSize;
- posBuf.put(x).put(y).put(z);
- }
- }
- if(data.normals != null) {
- // Unroll normals data array
- FloatBuffer normBuf = BufferUtils.createFloatBuffer(data.vCount * 3);
- mesh.setBuffer(VertexBuffer.Type.Normal, 3, normBuf);
- int[] mapping = null;
- if(data.normalsMapping.equals("ByVertice"))
- mapping = data.vertexMap;
- else if(data.normalsMapping.equals("ByPolygonVertex"))
- mapping = data.indexMap;
- int srcCount = data.normals.length / 3;
- for(int i = 0; i < data.vCount; ++i) {
- int index = mapping[i];
- if(index > srcCount)
- throw new AssetLoadException("Invalid normal mapping. Unexpected lookup normal " + index + " from " + srcCount);
- float x = (float) data.normals[3 * index + 0];
- float y = (float) data.normals[3 * index + 1];
- float z = (float) data.normals[3 * index + 2];
- normBuf.put(x).put(y).put(z);
- }
- }
- if(data.tangents != null) {
- // Unroll normals data array
- FloatBuffer tanBuf = BufferUtils.createFloatBuffer(data.vCount * 4);
- mesh.setBuffer(VertexBuffer.Type.Tangent, 4, tanBuf);
- int[] mapping = null;
- if(data.tangentsMapping.equals("ByVertice"))
- mapping = data.vertexMap;
- else if(data.tangentsMapping.equals("ByPolygonVertex"))
- mapping = data.indexMap;
- int srcCount = data.tangents.length / 3;
- for(int i = 0; i < data.vCount; ++i) {
- int index = mapping[i];
- if(index > srcCount)
- throw new AssetLoadException("Invalid tangent mapping. Unexpected lookup tangent " + index + " from " + srcCount);
- float x = (float) data.tangents[3 * index + 0];
- float y = (float) data.tangents[3 * index + 1];
- float z = (float) data.tangents[3 * index + 2];
- tanBuf.put(x).put(y).put(z).put(-1.0f);
- }
- }
- if(data.binormals != null) {
- // Unroll normals data array
- FloatBuffer binormBuf = BufferUtils.createFloatBuffer(data.vCount * 3);
- mesh.setBuffer(VertexBuffer.Type.Binormal, 3, binormBuf);
- int[] mapping = null;
- if(data.binormalsMapping.equals("ByVertice"))
- mapping = data.vertexMap;
- else if(data.binormalsMapping.equals("ByPolygonVertex"))
- mapping = data.indexMap;
- int srcCount = data.binormals.length / 3;
- for(int i = 0; i < data.vCount; ++i) {
- int index = mapping[i];
- if(index > srcCount)
- throw new AssetLoadException("Invalid binormal mapping. Unexpected lookup binormal " + index + " from " + srcCount);
- float x = (float) data.binormals[3 * index + 0];
- float y = (float) data.binormals[3 * index + 1];
- float z = (float) data.binormals[3 * index + 2];
- binormBuf.put(x).put(y).put(z);
- }
- }
- if(data.uv != null) {
- int[] unIndexMap = data.vertexMap;
- if(data.uvIndex != null) {
- int uvIndexSrcCount = data.uvIndex.length;
- if(uvIndexSrcCount != data.iCount)
- throw new AssetLoadException("Invalid number of texcoord index data " + uvIndexSrcCount + " expected " + data.iCount);
- // Unroll UV index array
- unIndexMap = new int[data.vCount];
- int n = 0;
- for(int i = 0; i < data.iCount; ++i) {
- int index = data.uvIndex[i];
- if(isQuads && (i % 4) == 3) {
- unIndexMap[n + 0] = data.uvIndex[i - 3];
- unIndexMap[n + 1] = data.uvIndex[i - 1];
- unIndexMap[n + 2] = index;
- n += 3;
- } else {
- unIndexMap[i] = index;
- }
- }
- }
- // Unroll UV data array
- FloatBuffer tcBuf = BufferUtils.createFloatBuffer(data.vCount * 2);
- mesh.setBuffer(VertexBuffer.Type.TexCoord, 2, tcBuf);
- int srcCount = data.uv.length / 2;
- for(int i = 0; i < data.vCount; ++i) {
- int index = unIndexMap[i];
- if(index > srcCount)
- throw new AssetLoadException("Invalid texcoord mapping. Unexpected lookup texcoord " + index + " from " + srcCount);
- float u = index >= 0 ? (float) data.uv[2 * index + 0] : 0;
- float v = index >= 0 ? (float) data.uv[2 * index + 1] : 0;
- tcBuf.put(u).put(v);
- }
- }
- mesh.setStatic();
- mesh.updateBound();
- mesh.updateCounts();
- Geometry geom = new Geometry();
- geom.setMesh(mesh);
- return geom;
- }
-
- private Material createMaterial(MaterialData data) {
- Material m = new Material(assetManager, "Common/MatDefs/Light/Lighting.j3md");
- m.setName(data.name);
- data.ambientColor.multLocal(data.ambientFactor);
- data.diffuseColor.multLocal(data.diffuseFactor);
- data.specularColor.multLocal(data.specularFactor);
- m.setColor("Ambient", new ColorRGBA(data.ambientColor.x, data.ambientColor.y, data.ambientColor.z, 1));
- m.setColor("Diffuse", new ColorRGBA(data.diffuseColor.x, data.diffuseColor.y, data.diffuseColor.z, 1));
- m.setColor("Specular", new ColorRGBA(data.specularColor.x, data.specularColor.y, data.specularColor.z, 1));
- m.setFloat("Shininess", data.shininessExponent);
- m.setBoolean("UseMaterialColors", true);
- return m;
- }
-
- private Node createNode(ModelData data) {
- Node model = new Node(data.name);
- model.setLocalTranslation(data.localTranslation);
- model.setLocalRotation(data.localRotation);
- model.setLocalScale(data.localScale);
- return model;
- }
-
- private Limb createLimb(ModelData data) {
- Limb limb = new Limb();
- limb.name = data.name;
- Quaternion rotation = data.preRotation.mult(data.localRotation);
- limb.bindTransform = new Transform(data.localTranslation, rotation, data.localScale);
- return limb;
- }
-
- private BindPose createPose(BindPoseData data) {
- BindPose pose = new BindPose();
- pose.name = data.name;
- for(NodeTransformData item : data.list) {
- Transform t = buildTransform(item.transform);
- //t.getTranslation().divideLocal(unitSize);
- t.getScale().multLocal(unitSize);
- pose.nodeTransforms.put(item.nodeId, t);
- }
- return pose;
- }
-
- private Texture createTexture(TextureData data) {
- Texture tex = new Texture2D();
- tex.setName(data.name);
- return tex;
- }
-
- private Image createImage(ImageData data) {
- Image image = null;
- if(data.filename != null) {
- // Try load by absolute path
- File file = new File(data.filename);
- if(file.exists() && file.isFile()) {
- File dir = new File(file.getParent());
- String locatorPath = dir.getAbsolutePath();
- Texture tex = null;
- try {
- assetManager.registerLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
- tex = assetManager.loadTexture(file.getName());
- } catch(Exception e) {
- } finally {
- assetManager.unregisterLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
- }
- if(tex != null)
- image = tex.getImage();
- }
- }
- if(image == null && data.relativeFilename != null) {
- // Try load by relative path
- File dir = new File(sceneFolderName);
- String locatorPath = dir.getAbsolutePath();
- Texture tex = null;
- try {
- assetManager.registerLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
- tex = assetManager.loadTexture(data.relativeFilename);
- } catch(Exception e) {
- } finally {
- assetManager.unregisterLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
- }
- if(tex != null)
- image = tex.getImage();
- }
- if(image == null && data.content != null) {
- // Try load from content
- String filename = null;
- if(data.filename != null)
- filename = new File(data.filename).getName();
- if(filename != null && data.relativeFilename != null)
- filename = data.relativeFilename;
- // Filename is required to aquire asset loader by extension
- if(filename != null) {
- String locatorPath = sceneFilename;
- filename = sceneFilename + File.separatorChar + filename; // Unique path
- Texture tex = null;
- try {
- assetManager.registerLocator(locatorPath, com.jme3.scene.plugins.fbx.ContentTextureLocator.class);
- tex = assetManager.loadTexture(new ContentTextureKey(filename, data.content));
- } catch(Exception e) {
- } finally {
- assetManager.unregisterLocator(locatorPath, com.jme3.scene.plugins.fbx.ContentTextureLocator.class);
+ obj = allObjects.get(objId);
+ ref = allObjects.get(refId);
+ if(ref != null)
+ ref.link(obj, propName);
+ break;
}
- if(tex != null)
- image = tex.getImage();
}
}
- if(image == null)
- throw new AssetLoadException("Content not loaded for image " + data.name);
- return image;
- }
-
- private Transform buildTransform(double[] transform) {
- float[] m = new float[transform.length];
- for(int i = 0; i < transform.length; ++i)
- m[i] = (float) transform[i];
- Matrix4f matrix = new Matrix4f(m);
- Vector3f pos = matrix.toTranslationVector();
- Quaternion rot = matrix.toRotationQuat();
- Vector3f scale = matrix.toScaleVector();
- return new Transform(pos, rot, scale);
- }
-
- private Quaternion quatFromBoneAngles(float xAngle, float yAngle, float zAngle) {
- float angle;
- float sinY, sinZ, sinX, cosY, cosZ, cosX;
- angle = zAngle * 0.5f;
- sinZ = FastMath.sin(angle);
- cosZ = FastMath.cos(angle);
- angle = yAngle * 0.5f;
- sinY = FastMath.sin(angle);
- cosY = FastMath.cos(angle);
- angle = xAngle * 0.5f;
- sinX = FastMath.sin(angle);
- cosX = FastMath.cos(angle);
- float cosYXcosZ = cosY * cosZ;
- float sinYXsinZ = sinY * sinZ;
- float cosYXsinZ = cosY * sinZ;
- float sinYXcosZ = sinY * cosZ;
- // For some reason bone space is differ, this is modified formulas
- float w = (cosYXcosZ * cosX + sinYXsinZ * sinX);
- float x = (cosYXcosZ * sinX - sinYXsinZ * cosX);
- float y = (sinYXcosZ * cosX + cosYXsinZ * sinX);
- float z = (cosYXsinZ * cosX - sinYXcosZ * sinX);
- return new Quaternion(x, y, z, w).normalizeLocal();
}
- private Node linkScene() {
+ private void linkScene() {
logger.log(Level.FINE, "Linking scene objects");
long startTime = System.currentTimeMillis();
- Node sceneNode = linkSceneNodes();
- linkMaterials();
- linkMeshes(sceneNode);
- linkSkins(sceneNode);
- linkAnimations();
+ applySkinning();
+ buildAnimations();
+ for(FbxMesh mesh : geomMap.values())
+ mesh.clearMaterials();
+ // Remove bones from node structures : JME creates attach node by itself
+ for(FbxNode limb : limbMap.values())
+ limb.node.removeFromParent();
long estimatedTime = System.currentTimeMillis() - startTime;
logger.log(Level.FINE, "Linking done in {0} ms", estimatedTime);
- return sceneNode;
}
- private Node linkSceneNodes() {
- Node sceneNode = new Node(sceneName + "-scene");
- // Build mesh nodes
- for(long nodeId : modelDataMap.keySet()) {
- ModelData data = modelDataMap.get(nodeId);
- Node node = createNode(data);
- modelMap.put(nodeId, node);
- }
- // Link model nodes into scene
- for(long modelId : modelMap.keySet()) {
- List refs = refMap.get(modelId);
- if(refs == null)
- continue;
- Node model = modelMap.get(modelId);
- for(long refId : refs) {
- Node rootNode = (refId != 0) ? modelMap.get(refId) : sceneNode;
- if(rootNode != null)
- rootNode.attachChild(model);
- }
- }
- // Build bind poses
- for(long poseId : poseDataMap.keySet()) {
- BindPoseData data = poseDataMap.get(poseId);
- BindPose pose = createPose(data);
- if(pose != null)
- bindMap.put(poseId, pose);
- }
- // Apply bind poses
- for(BindPose pose : bindMap.values()) {
- for(long nodeId : pose.nodeTransforms.keySet()) {
- Node model = modelMap.get(nodeId);
- if(model != null) {
- Transform t = pose.nodeTransforms.get(nodeId);
- model.setLocalTransform(t);
- }
- }
- }
- return sceneNode;
- }
-
- private void linkMaterials() {
- // Build materials
- for(long matId : matDataMap.keySet()) {
- MaterialData data = matDataMap.get(matId);
- Material material = createMaterial(data);
- if(material != null)
- matMap.put(matId, material);
- }
- // Build textures
- for(long texId : texDataMap.keySet()) {
- TextureData data = texDataMap.get(texId);
- Texture tex = createTexture(data);
- if(tex != null)
- texMap.put(texId, tex);
- }
- // Build Images
- for(long imgId : imgDataMap.keySet()) {
- ImageData data = imgDataMap.get(imgId);
- Image img = createImage(data);
- if(img != null)
- imgMap.put(imgId, img);
- }
- // Link images to textures
- for(long imgId : imgMap.keySet()) {
- List refs = refMap.get(imgId);
- if(refs == null)
- continue;
- Image img = imgMap.get(imgId);
- for(long refId : refs) {
- Texture tex = texMap.get(refId);
- if(tex != null)
- tex.setImage(img);
- }
- }
- // Link textures to material maps
- for(long texId : texMap.keySet()) {
- List props = propMap.get(texId);
- if(props == null)
- continue;
- Texture tex = texMap.get(texId);
- for(PropertyLink prop : props) {
- Material mat = matMap.get(prop.ref);
- if(mat != null) {
- if(prop.propName.equals("DiffuseColor")) {
- mat.setTexture("DiffuseMap", tex);
- mat.setColor("Diffuse", ColorRGBA.White);
- } else if(prop.propName.equals("SpecularColor")) {
- mat.setTexture("SpecularMap", tex);
- mat.setColor("Specular", ColorRGBA.White);
- } else if(prop.propName.equals("NormalMap"))
- mat.setTexture("NormalMap", tex);
- }
- }
- }
- }
-
- private void linkMeshes(Node sceneNode) {
- // Build meshes
- for(long meshId : meshDataMap.keySet()) {
- MeshData data = meshDataMap.get(meshId);
- Geometry geom = createGeomerty(data);
- if(geom != null)
- geomMap.put(meshId, geom);
- }
- // Link meshes to models
- for(long geomId : geomMap.keySet()) {
- List refs = refMap.get(geomId);
- if(refs == null)
- continue;
- Geometry geom = geomMap.get(geomId);
- for(long refId : refs) {
- Node rootNode = (refId != 0) ? modelMap.get(refId) : sceneNode;
- if(rootNode != null) {
- geom.setName(rootNode.getName() + "-mesh");
- geom.updateModelBound();
- rootNode.attachChild(geom);
- break;
- }
- }
- }
- // Link materials to meshes
- for(long matId : matMap.keySet()) {
- List refs = refMap.get(matId);
- if(refs == null)
- continue;
- Material mat = matMap.get(matId);
- for(long refId : refs) {
- Node rootNode = modelMap.get(refId);
- if(rootNode != null) {
- for(Spatial child : rootNode.getChildren())
- child.setMaterial(mat);
- }
- }
- }
- }
-
- private void linkSkins(Node sceneNode) {
- // Build skeleton limbs
- for(long nodeId : modelDataMap.keySet()) {
- ModelData data = modelDataMap.get(nodeId);
- if(data.type.equals("LimbNode")) {
- Limb limb = createLimb(data);
- limbMap.put(nodeId, limb);
- }
- }
- if(limbMap.size() == 0)
+ private void applySkinning() {
+ for(FbxBindPose pose : bindMap.values())
+ pose.fillBindTransforms();
+ if(limbMap == null)
return;
- // Build skeleton bones
- Map bones = new HashMap();
- for(long limbId : limbMap.keySet()) {
- Limb limb = limbMap.get(limbId);
- Bone bone = new Bone(limb.name);
- Transform t = limb.bindTransform;
- bone.setBindTransforms(t.getTranslation(), t.getRotation(), t.getScale());
- bones.put(limbId, bone);
- }
- // Attach bones to roots
- for(long limbId : limbMap.keySet()) {
- List refs = refMap.get(limbId);
- if(refs == null)
- continue;
- // Find root limb
- long rootLimbId = 0L;
- for(long refId : refs) {
- if(limbMap.containsKey(refId)) {
- rootLimbId = refId;
- break;
- }
- }
- if(rootLimbId != 0L) {
- Bone bone = bones.get(limbId);
- Bone root = bones.get(rootLimbId);
- root.addChild(bone);
- }
- }
- // Link bone clusters to skin
- for(long clusterId : clusterMap.keySet()) {
- List refs = refMap.get(clusterId);
- if(refs == null)
- continue;
- for(long skinId : refs) {
- if(skinMap.containsKey(skinId)) {
- ClusterData data = clusterMap.get(clusterId);
- data.skinId = skinId;
- break;
- }
+ List bones = new ArrayList<>();
+ for(FbxNode limb : limbMap.values()) {
+ if(limb.bone != null) {
+ bones.add(limb.bone);
+ limb.buildBindPoseBoneTransform();
}
}
- // Build the skeleton
- this.skeleton = new Skeleton(bones.values().toArray(new Bone[0]));
+ skeleton = new Skeleton(bones.toArray(new Bone[bones.size()]));
skeleton.setBindingPose();
- for(long limbId : bones.keySet()) {
- Bone bone = bones.get(limbId);
- Limb limb = limbMap.get(limbId);
- limb.boneIndex = skeleton.getBoneIndex(bone);
- }
- // Assign bones skinning to meshes
- for(long skinId : skinMap.keySet()) {
- // Find mesh by skin
- Mesh mesh = null;
- MeshData meshData = null;
- for(long meshId : refMap.get(skinId)) {
- Geometry g = geomMap.get(meshId);
- if(g != null) {
- meshData = meshDataMap.get(meshId);
- mesh = g.getMesh();
- break;
- }
- }
- // Bind skinning indexes and weight
- if(mesh != null && meshData != null) {
- // Create bone buffers
- FloatBuffer boneWeightData = BufferUtils.createFloatBuffer(meshData.vCount * 4);
- ByteBuffer boneIndicesData = BufferUtils.createByteBuffer(meshData.vCount * 4);
- mesh.setBuffer(VertexBuffer.Type.BoneWeight, 4, boneWeightData);
- mesh.setBuffer(VertexBuffer.Type.BoneIndex, 4, boneIndicesData);
- mesh.getBuffer(VertexBuffer.Type.BoneWeight).setUsage(Usage.CpuOnly);
- mesh.getBuffer(VertexBuffer.Type.BoneIndex).setUsage(Usage.CpuOnly);
- VertexBuffer weightsHW = new VertexBuffer(Type.HWBoneWeight);
- VertexBuffer indicesHW = new VertexBuffer(Type.HWBoneIndex);
- indicesHW.setUsage(Usage.CpuOnly); // Setting usage to CpuOnly so that the buffer is not send empty to the GPU
- weightsHW.setUsage(Usage.CpuOnly);
- mesh.setBuffer(weightsHW);
- mesh.setBuffer(indicesHW);
- // Accumulate skin bones influence into mesh buffers
- boolean bonesLimitExceeded = false;
- for(long limbId : bones.keySet()) {
- // Search bone cluster for the given limb and skin
- ClusterData cluster = null;
- for(long clusterId : refMap.get(limbId)) {
- ClusterData data = clusterMap.get(clusterId);
- if(data != null && data.skinId == skinId) {
- cluster = data;
- break;
- }
- }
- if(cluster == null || cluster.indexes == null || cluster.weights == null || cluster.indexes.length != cluster.weights.length)
- continue;
- Limb limb = limbMap.get(limbId);
- if(limb.boneIndex > 255)
- throw new AssetLoadException("Bone index can't be packed into byte");
- for(int i = 0; i < cluster.indexes.length; ++i) {
- int vertexIndex = cluster.indexes[i];
- if(vertexIndex >= meshData.reverseVertexMap.size())
- throw new AssetLoadException("Invalid skinning vertex index. Unexpected index lookup " + vertexIndex + " from " + meshData.reverseVertexMap.size());
- List dstVertices = meshData.reverseVertexMap.get(vertexIndex);
- for(int v : dstVertices) {
- // Append bone index and weight to vertex
- int offset;
- float w = 0;
- for(offset = v * 4; offset < v * 4 + 4; ++offset) {
- w = boneWeightData.get(offset);
- if(w == 0)
- break;
- }
- if(w == 0) {
- boneWeightData.put(offset, (float) cluster.weights[i]);
- boneIndicesData.put(offset, (byte) limb.boneIndex);
- } else {
- // TODO It would be nice to discard small weight to accumulate more heavy influence
- bonesLimitExceeded = true;
- }
- }
- }
- }
- if(bonesLimitExceeded)
- logger.log(Level.WARNING, "Skinning support max 4 bone per vertex. Exceeding data will be discarded.");
- // Postprocess bones weights
- int maxWeightsPerVert = 0;
- boneWeightData.rewind();
- for(int v = 0; v < meshData.vCount; v++) {
- float w0 = boneWeightData.get();
- float w1 = boneWeightData.get();
- float w2 = boneWeightData.get();
- float w3 = boneWeightData.get();
- if(w3 != 0) {
- maxWeightsPerVert = Math.max(maxWeightsPerVert, 4);
- } else if(w2 != 0) {
- maxWeightsPerVert = Math.max(maxWeightsPerVert, 3);
- } else if(w1 != 0) {
- maxWeightsPerVert = Math.max(maxWeightsPerVert, 2);
- } else if(w0 != 0) {
- maxWeightsPerVert = Math.max(maxWeightsPerVert, 1);
- }
- float sum = w0 + w1 + w2 + w3;
- if(sum != 1f) {
- // normalize weights
- float mult = (sum != 0) ? (1f / sum) : 0;
- boneWeightData.position(v * 4);
- boneWeightData.put(w0 * mult);
- boneWeightData.put(w1 * mult);
- boneWeightData.put(w2 * mult);
- boneWeightData.put(w3 * mult);
- }
- }
- mesh.setMaxNumWeights(maxWeightsPerVert);
- mesh.generateBindPose(true);
- }
- }
+ for(FbxNode limb : limbMap.values())
+ limb.setSkeleton(skeleton);
+ for(FbxSkin skin : skinMap.values())
+ skin.generateSkinning();
// Attach controls
animControl = new AnimControl(skeleton);
sceneNode.addControl(animControl);
@@ -1258,45 +299,20 @@ public class SceneLoader implements AssetLoader {
sceneNode.addControl(control);
}
- private void linkAnimations() {
+ private void buildAnimations() {
if(skeleton == null)
return;
- if(animList == null || animList.list.size() == 0)
- return;
- // Link curves to nodes
- for(long curveId : acurveMap.keySet()) {
- List props = propMap.get(curveId);
- if(props == null)
- continue;
- AnimCurveData acurve = acurveMap.get(curveId);
- for(PropertyLink prop : props) {
- AnimNode anode = anodeMap.get(prop.ref);
- if(anode != null) {
- if(prop.propName.equals("d|X"))
- anode.xCurve = acurve;
- else if(prop.propName.equals("d|Y"))
- anode.yCurve = acurve;
- else if(prop.propName.equals("d|Z"))
- anode.zCurve = acurve;
- }
- }
- }
- // Link nodes to layers
- for(long nodeId : anodeMap.keySet()) {
- List refs = refMap.get(nodeId);
- if(refs == null)
- continue;
- for(long layerId : refs) {
- if(alayerMap.containsKey(layerId)) {
- AnimNode anode = anodeMap.get(nodeId);
- anode.layerId = layerId;
- break;
- }
+ if(animList == null || animList.list.size() == 0) {
+ animList = new AnimationList();
+ for(long layerId : alayerMap.keySet()) {
+ FbxObject layer = alayerMap.get(layerId);
+ animList.add(layer.name, layer.name, 0, -1);
}
}
// Extract aminations
HashMap anims = new HashMap();
for(AnimInverval animInfo : animList.list) {
+ float realLength = 0;
float length = (animInfo.lastFrame - animInfo.firstFrame) / this.animFrameRate;
float animStart = animInfo.firstFrame / this.animFrameRate;
float animStop = animInfo.lastFrame / this.animFrameRate;
@@ -1304,65 +320,50 @@ public class SceneLoader implements AssetLoader {
// Search source layer for animation nodes
long sourceLayerId = 0L;
for(long layerId : alayerMap.keySet()) {
- AnimLayer layer = alayerMap.get(layerId);
+ FbxObject layer = alayerMap.get(layerId);
if(layer.name.equals(animInfo.layerName)) {
sourceLayerId = layerId;
break;
}
}
- // Assign animation nodes to limbs
- for(Limb limb : limbMap.values()) {
- limb.animTranslation = null;
- limb.animRotation = null;
- limb.animScale = null;
- }
- for(long nodeId : anodeMap.keySet()) {
- List props = propMap.get(nodeId);
- if(props == null)
- continue;
- AnimNode anode = anodeMap.get(nodeId);
- if(sourceLayerId != 0L && anode.layerId != sourceLayerId)
- continue; // filter node
- for(PropertyLink prop : props) {
- Limb limb = limbMap.get(prop.ref);
- if(limb != null) {
- if(prop.propName.equals("Lcl Translation"))
- limb.animTranslation = anode;
- else if(prop.propName.equals("Lcl Rotation"))
- limb.animRotation = anode;
- else if(prop.propName.equals("Lcl Scaling"))
- limb.animScale = anode;
- }
- }
- }
// Build bone tracks
- for(Limb limb : limbMap.values()) {
- long[] keyTimes = null;
- boolean haveTranslation = (limb.animTranslation != null && limb.animTranslation.xCurve != null && limb.animTranslation.yCurve != null && limb.animTranslation.zCurve != null);
- boolean haveRotation = (limb.animRotation != null && limb.animRotation.xCurve != null && limb.animRotation.yCurve != null && limb.animRotation.zCurve != null);
- boolean haveScale = (limb.animScale != null && limb.animScale.xCurve != null && limb.animScale.yCurve != null && limb.animScale.zCurve != null);
- // Search key time array
+ for(FbxNode limb : limbMap.values()) {
+ // Animation channels may have different keyframes (non-baked animation).
+ // So we have to restore intermediate values for all channels cause of JME requires
+ // a bone track as a single channel with collective transformation for each keyframe
+ Set stamps = new TreeSet(); // Sorted unique timestamps
+ FbxAnimNode animTranslation = limb.animTranslation(sourceLayerId);
+ FbxAnimNode animRotation = limb.animRotation(sourceLayerId);
+ FbxAnimNode animScale = limb.animScale(sourceLayerId);
+ boolean haveTranslation = haveAnyChannel(animTranslation);
+ boolean haveRotation = haveAnyChannel(animRotation);
+ boolean haveScale = haveAnyChannel(animScale);
+ // Collect keyframes stamps
if(haveTranslation)
- keyTimes = limb.animTranslation.xCurve.keyTimes;
- else if(haveRotation)
- keyTimes = limb.animRotation.xCurve.keyTimes;
- else if(haveScale)
- keyTimes = limb.animScale.xCurve.keyTimes;
- if(keyTimes == null)
+ animTranslation.exportTimes(stamps);
+ if(haveRotation)
+ animRotation.exportTimes(stamps);
+ if(haveScale)
+ animScale.exportTimes(stamps);
+ if(stamps.isEmpty())
continue;
+ long[] keyTimes = new long[stamps.size()];
+ int cnt = 0;
+ for(long t : stamps)
+ keyTimes[cnt++] = t;
// Calculate keys interval by animation time interval
- int firstKey = 0;
- int lastKey = keyTimes.length - 1;
+ int firstKeyIndex = 0;
+ int lastKeyIndex = keyTimes.length - 1;
for(int i = 0; i < keyTimes.length; ++i) {
float time = (float) (((double) keyTimes[i]) * secondsPerUnit); // Translate into seconds
if(time <= animStart)
- firstKey = i;
- if(time >= animStop) {
- lastKey = i;
+ firstKeyIndex = i;
+ if(time >= animStop && animStop >= 0) {
+ lastKeyIndex = i;
break;
}
}
- int keysCount = lastKey - firstKey + 1;
+ int keysCount = lastKeyIndex - firstKeyIndex + 1;
if(keysCount <= 0)
continue;
float[] times = new float[keysCount];
@@ -1371,43 +372,42 @@ public class SceneLoader implements AssetLoader {
Vector3f[] scales = null;
// Calculate keyframes times
for(int i = 0; i < keysCount; ++i) {
- int keyIndex = firstKey + i;
+ int keyIndex = firstKeyIndex + i;
float time = (float) (((double) keyTimes[keyIndex]) * secondsPerUnit); // Translate into seconds
times[i] = time - animStart;
+ realLength = Math.max(realLength, times[i]);
}
// Load keyframes from animation curves
if(haveTranslation) {
for(int i = 0; i < keysCount; ++i) {
- int keyIndex = firstKey + i;
- float x = limb.animTranslation.xCurve.keyValues[keyIndex] - limb.animTranslation.value.x;
- float y = limb.animTranslation.yCurve.keyValues[keyIndex] - limb.animTranslation.value.y;
- float z = limb.animTranslation.zCurve.keyValues[keyIndex] - limb.animTranslation.value.z;
- translations[i] = new Vector3f(x, y, z).divideLocal(unitSize);
+ int keyIndex = firstKeyIndex + i;
+ FbxAnimNode n = animTranslation;
+ Vector3f tvec = n.getValue(keyTimes[keyIndex], n.value).subtractLocal(n.value); // Why do it here but not in other places? FBX magic?
+ translations[i] = tvec.divideLocal(unitSize);
}
} else {
for(int i = 0; i < keysCount; ++i)
- translations[i] = new Vector3f();
+ translations[i] = Vector3f.ZERO;
}
+ RotationOrder ro = RotationOrder.EULER_XYZ;
if(haveRotation) {
for(int i = 0; i < keysCount; ++i) {
- int keyIndex = firstKey + i;
- float x = limb.animRotation.xCurve.keyValues[keyIndex];
- float y = limb.animRotation.yCurve.keyValues[keyIndex];
- float z = limb.animRotation.zCurve.keyValues[keyIndex];
- rotations[i] = new Quaternion().fromAngles(FastMath.DEG_TO_RAD * x, FastMath.DEG_TO_RAD * y, FastMath.DEG_TO_RAD * z);
+ int keyIndex = firstKeyIndex + i;
+ FbxAnimNode n = animRotation;
+ Vector3f tvec = n.getValue(keyTimes[keyIndex], n.value);
+ rotations[i] = ro.rotate(tvec);
}
} else {
for(int i = 0; i < keysCount; ++i)
- rotations[i] = new Quaternion();
+ rotations[i] = Quaternion.IDENTITY;
}
if(haveScale) {
scales = new Vector3f[keysCount];
for(int i = 0; i < keysCount; ++i) {
- int keyIndex = firstKey + i;
- float x = limb.animScale.xCurve.keyValues[keyIndex];
- float y = limb.animScale.yCurve.keyValues[keyIndex];
- float z = limb.animScale.zCurve.keyValues[keyIndex];
- scales[i] = new Vector3f(x, y, z);
+ int keyIndex = firstKeyIndex + i;
+ FbxAnimNode n = animScale;
+ Vector3f tvec = n.getValue(keyTimes[keyIndex], n.value);
+ scales[i] = tvec;
}
}
BoneTrack track = null;
@@ -1417,166 +417,59 @@ public class SceneLoader implements AssetLoader {
track = new BoneTrack(limb.boneIndex, times, translations, rotations);
anim.addTrack(track);
}
+ if(realLength != length && animInfo.lastFrame == -1) {
+ Track[] tracks = anim.getTracks();
+ if(tracks == null || tracks.length == 0)
+ continue;
+ anim = new Animation(animInfo.name, realLength);
+ for(Track track : tracks)
+ anim.addTrack(track);
+ }
anims.put(anim.getName(), anim);
}
animControl.setAnimations(anims);
}
private void releaseObjects() {
- meshDataMap.clear();
- matDataMap.clear();
- texDataMap.clear();
- imgDataMap.clear();
- modelDataMap.clear();
- poseDataMap.clear();
+ // Reset settings
+ unitSize = 1;
+ animFrameRate = 30;
+ xAxis = 1;
+ yAxis = 1;
+ zAxis = 1;
+ // Clear cache
+ warnings.clear();
+ animList = null;
+ sceneName = null;
+ sceneFilename = null;
+ sceneFolderName = null;
+ assetManager = null;
+ currentAssetInfo = null;
+ // Clear objects
+ allObjects.clear();
skinMap.clear();
- clusterMap.clear();
- acurveMap.clear();
- anodeMap.clear();
alayerMap.clear();
- refMap.clear();
- propMap.clear();
modelMap.clear();
limbMap.clear();
bindMap.clear();
geomMap.clear();
- matMap.clear();
- texMap.clear();
- imgMap.clear();
skeleton = null;
animControl = null;
- animList = null;
- }
-
- private class MeshData {
- double[] vertices;
- int[] indices;
- int[] edges;
- String normalsMapping;
- String normalsReference;
- double[] normals;
- String tangentsMapping;
- String tangentsReference;
- double[] tangents;
- String binormalsMapping;
- String binormalsReference;
- double[] binormals;
- String uvMapping;
- String uvReference;
- double[] uv;
- int[] uvIndex;
- String smoothingMapping;
- String smoothingReference;
- int[] smoothing;
- String materialsMapping;
- String materialsReference;
- int[] materials;
- // Build helping data
- int iCount;
- int vCount;
- int srcVertexCount;
- int[] vertexMap; // Target vertex -> source vertex
- List> reverseVertexMap; // source vertex -> list of target vertices
- int[] indexMap; // Target vertex -> source index
- }
-
- private class ModelData {
- String name;
- String type;
- Vector3f localTranslation = new Vector3f();
- Quaternion localRotation = new Quaternion();
- Vector3f localScale = new Vector3f(Vector3f.UNIT_XYZ);
- Quaternion preRotation = new Quaternion();
+ sceneNode = null;
}
- private class NodeTransformData {
- long nodeId;
- double[] transform;
- }
- private class BindPoseData {
- String name;
- List list = new LinkedList();
+ private static boolean haveAnyChannel(FbxAnimNode anims) {
+ return anims != null && anims.haveAnyChannel();
}
- private class BindPose {
- String name;
- Map nodeTransforms = new HashMap();
- }
-
- private class MaterialData {
- String name;
- String shadingModel = "phong";
- Vector3f ambientColor = new Vector3f(0.2f, 0.2f, 0.2f);
- float ambientFactor = 1.0f;
- Vector3f diffuseColor = new Vector3f(0.8f, 0.8f, 0.8f);
- float diffuseFactor = 1.0f;
- Vector3f specularColor = new Vector3f(0.2f, 0.2f, 0.2f);
- float specularFactor = 1.0f;
- float shininessExponent = 20.0f;
- }
-
- private class TextureData {
- String name;
- String bindType;
- String filename;
- }
-
- private class ImageData {
- String name;
- String type;
- String filename;
- String relativeFilename;
- byte[] content;
- }
-
- private class Limb {
- String name;
- Transform bindTransform;
- int boneIndex;
- AnimNode animTranslation;
- AnimNode animRotation;
- AnimNode animScale;
- }
-
- private class ClusterData {
- int[] indexes;
- double[] weights;
- double[] transform;
- double[] transformLink;
- long skinId;
- }
-
- private class SkinData {
- String type;
- }
-
- private class AnimCurveData {
- long[] keyTimes;
- float[] keyValues;
- }
-
- private class AnimLayer {
- String name;
- }
-
- private class AnimNode {
- String name;
- Vector3f value;
- AnimCurveData xCurve;
- AnimCurveData yCurve;
- AnimCurveData zCurve;
- long layerId;
- }
-
- private class PropertyLink {
- long ref;
- String propName;
-
- public PropertyLink(long id, String prop) {
- this.ref = id;
- this.propName = prop;
+ private static String join(List list, String glue) {
+ StringBuilder sb = new StringBuilder();
+ for(int i = 0; i < list.size(); ++i) {
+ if(sb.length() != 0)
+ sb.append(glue);
+ sb.append(list.get(i));
}
+ return sb.toString();
}
-
}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxAnimCurve.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxAnimCurve.java
new file mode 100644
index 000000000..395216156
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxAnimCurve.java
@@ -0,0 +1,49 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxAnimCurve extends FbxObject {
+
+ public long[] keyTimes;
+ public float[] keyValues;
+ public float defaultValue = 0.0f;
+
+ public FbxAnimCurve(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ for(FbxElement e : element.children) {
+ switch(e.id) {
+ case "KeyTime":
+ keyTimes = (long[]) e.properties.get(0);
+ break;
+ case "KeyValueFloat":
+ keyValues = (float[]) e.properties.get(0);
+ break;
+ case "Default":
+ defaultValue = ((Number) e.properties.get(0)).floatValue();
+ break;
+ }
+ }
+ }
+
+ public float getValue(long time) {
+ // Search animation interval
+ for(int i = 0; i < keyTimes.length; ++i) {
+ if(keyTimes[i] == time) { // hit the keyframe
+ return keyValues[i];
+ } else if(keyTimes[i] > time) {
+ if(i == 0) { // left from the whole range
+ return defaultValue;//keyValues[0];
+ } else {
+ // Interpolate between two keyframes
+ float dt = (float) (keyTimes[i] - keyTimes[i - 1]);
+ float dtInt = (float) (time - keyTimes[i - 1]);
+ float dv = keyValues[i] - keyValues[i - 1];
+ return keyValues[i - 1] + dv * (dtInt / dt);
+ }
+ }
+ }
+ // right from the whole range
+ return defaultValue;//keyValues[keyValues.length - 1];
+ }
+}
\ No newline at end of file
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxAnimNode.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxAnimNode.java
new file mode 100644
index 000000000..7febc92d1
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxAnimNode.java
@@ -0,0 +1,86 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import java.util.Collection;
+
+import com.jme3.math.Vector3f;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxAnimNode extends FbxObject {
+
+ public Vector3f value;
+ public FbxAnimCurve xCurve;
+ public FbxAnimCurve yCurve;
+ public FbxAnimCurve zCurve;
+ public long layerId;
+
+ public FbxAnimNode(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ if(type.equals("")) {
+ Double x = null, y = null, z = null;
+ for(FbxElement e2 : element.getFbxProperties()) {
+ String propName = (String) e2.properties.get(0);
+ switch(propName) {
+ case "d|X":
+ x = (Double) e2.properties.get(4);
+ break;
+ case "d|Y":
+ y = (Double) e2.properties.get(4);
+ break;
+ case "d|Z":
+ z = (Double) e2.properties.get(4);
+ break;
+ }
+ }
+ // Load only T R S curve nodes
+ if(x != null && y != null && z != null)
+ value = new Vector3f(x.floatValue(), y.floatValue(), z.floatValue());
+ }
+ }
+
+ @Override
+ public void link(FbxObject otherObject, String propertyName) {
+ if(otherObject instanceof FbxAnimCurve) {
+ FbxAnimCurve curve = (FbxAnimCurve) otherObject;
+ switch(propertyName) {
+ case "d|X":
+ xCurve = curve;
+ break;
+ case "d|Y":
+ yCurve = curve;
+ break;
+ case "d|Z":
+ zCurve = curve;
+ break;
+ }
+ }
+ }
+
+ @Override
+ public void link(FbxObject otherObject) {
+ layerId = otherObject.id;
+ }
+
+ public boolean haveAnyChannel() {
+ return xCurve != null || yCurve != null || zCurve != null;
+ }
+
+ public void exportTimes(Collection stamps) {
+ if(xCurve != null)
+ for(long t : xCurve.keyTimes)
+ stamps.add(t);
+ if(yCurve != null)
+ for(long t : yCurve.keyTimes)
+ stamps.add(t);
+ if(zCurve != null)
+ for(long t : zCurve.keyTimes)
+ stamps.add(t);
+ }
+
+ public Vector3f getValue(long time, Vector3f defaultValue) {
+ float xValue = (xCurve != null) ? xCurve.getValue(time) : defaultValue.x;
+ float yValue = (yCurve != null) ? yCurve.getValue(time) : defaultValue.y;
+ float zValue = (zCurve != null) ? zCurve.getValue(time) : defaultValue.z;
+ return new Vector3f(xValue, yValue, zValue);
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxBindPose.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxBindPose.java
new file mode 100644
index 000000000..1170f9718
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxBindPose.java
@@ -0,0 +1,55 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import com.jme3.math.Matrix4f;
+import com.jme3.math.Vector3f;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxBindPose extends FbxObject {
+
+ public Map nodeTransforms = new HashMap<>();
+
+ public FbxBindPose(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ if(type.equals("BindPose")) {
+ for(FbxElement e : element.children) {
+ if(e.id.equals("PoseNode")) {
+ long nodeId = 0;
+ double[] transform = null;
+ for(FbxElement e2 : e.children) {
+ switch(e2.id) {
+ case "Node":
+ nodeId = (Long) e2.properties.get(0);
+ break;
+ case "Matrix":
+ transform = (double[]) e2.properties.get(0);
+ break;
+ }
+ }
+ Matrix4f t = buildTransform(transform);
+ t.scale(new Vector3f(scene.unitSize, scene.unitSize, scene.unitSize));
+ nodeTransforms.put(nodeId, t);
+ }
+ }
+ }
+ }
+
+ public void fillBindTransforms() {
+ for(long nodeId : nodeTransforms.keySet()) {
+ FbxNode node = scene.modelMap.get(nodeId);
+ node.bindTransform = nodeTransforms.get(nodeId).clone();
+ }
+ }
+
+ private static Matrix4f buildTransform(double[] transform) {
+ float[] m = new float[transform.length];
+ for(int i = 0; i < transform.length; ++i)
+ m[i] = (float) transform[i];
+ Matrix4f matrix = new Matrix4f();
+ matrix.set(m, false);
+ return matrix;
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxCluster.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxCluster.java
new file mode 100644
index 000000000..354354567
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxCluster.java
@@ -0,0 +1,42 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxCluster extends FbxObject {
+
+ public int[] indexes;
+ public double[] weights;
+ public double[] transform;
+ public double[] transformLink;
+ public FbxSkin skin;
+
+ public FbxCluster(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ for(FbxElement e : element.children) {
+ switch(e.id) {
+ case "Indexes":
+ indexes = (int[]) e.properties.get(0);
+ break;
+ case "Weights":
+ weights = (double[]) e.properties.get(0);
+ break;
+ case "Transform":
+ transform = (double[]) e.properties.get(0);
+ break;
+ case "TransformLink":
+ transformLink = (double[]) e.properties.get(0);
+ break;
+ }
+ }
+ }
+
+ @Override
+ public void link(FbxObject child) {
+ if(child instanceof FbxNode) {
+ FbxNode limb = (FbxNode) child;
+ limb.skinToCluster.put(skin.id, this);
+ skin.bones.add(limb);
+ }
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxImage.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxImage.java
new file mode 100644
index 000000000..0b50aa163
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxImage.java
@@ -0,0 +1,123 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import java.io.File;
+
+import com.jme3.asset.AssetManager;
+import com.jme3.texture.Image;
+import com.jme3.texture.Texture;
+import com.jme3.texture.image.ColorSpace;
+import com.jme3.util.BufferUtils;
+import com.jme3.scene.plugins.fbx.ContentTextureKey;
+import com.jme3.scene.plugins.fbx.ContentTextureLocator;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxImage extends FbxObject {
+
+ String filename;
+ String relativeFilename;
+ byte[] content;
+ String imageType;
+
+ public Image image;
+
+ public FbxImage(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ if(type.equals("Clip")) {
+ for(FbxElement e : element.children) {
+ switch(e.id) {
+ case "Type":
+ imageType = (String) e.properties.get(0);
+ break;
+ case "Filename":
+ case "FileName":
+ filename = (String) e.properties.get(0);
+ break;
+ case "RelativeFilename":
+ relativeFilename = (String) e.properties.get(0);
+ break;
+ case "Content":
+ if(e.properties.size() > 0)
+ content = (byte[]) e.properties.get(0);
+ break;
+ }
+ }
+ image = createImage();
+ }
+ }
+
+
+ private Image createImage() {
+ AssetManager assetManager = scene.assetManager;
+ Image image = null;
+ if(filename != null) {
+ // Try load by absolute path
+ File file = new File(filename);
+ if(file.exists() && file.isFile()) {
+ File dir = new File(file.getParent());
+ String locatorPath = dir.getAbsolutePath();
+ Texture tex = null;
+ try {
+ assetManager.registerLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
+ tex = assetManager.loadTexture(file.getName());
+ } catch(Exception e) {} finally {
+ assetManager.unregisterLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
+ }
+ if(tex != null)
+ image = tex.getImage();
+ }
+ }
+ if(image == null && relativeFilename != null) {
+ // Try load by relative path
+ File dir = new File(scene.sceneFolderName);
+ String locatorPath = dir.getAbsolutePath();
+ Texture tex = null;
+ try {
+ assetManager.registerLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
+ tex = assetManager.loadTexture(relativeFilename);
+ } catch(Exception e) {} finally {
+ assetManager.unregisterLocator(locatorPath, com.jme3.asset.plugins.FileLocator.class);
+ }
+ if(tex != null)
+ image = tex.getImage();
+ }
+ if(image == null && content != null) {
+ // Try load from content
+ String filename = null;
+ if(this.filename != null)
+ filename = new File(this.filename).getName();
+ if(filename != null && this.relativeFilename != null)
+ filename = this.relativeFilename;
+ // Filename is required to aquire asset loader by extension
+ if(filename != null) {
+ String locatorPath = scene.sceneFilename;
+ filename = scene.sceneFilename + File.separatorChar + filename; // Unique path
+ Texture tex = null;
+ try {
+ assetManager.registerLocator(locatorPath, ContentTextureLocator.class);
+ tex = assetManager.loadTexture(new ContentTextureKey(filename, content));
+ } catch(Exception e) {} finally {
+ assetManager.unregisterLocator(locatorPath, ContentTextureLocator.class);
+ }
+ if(tex != null)
+ image = tex.getImage();
+ }
+ }
+ if(image == null) {
+ // Try to load from files near
+ if(relativeFilename != null) {
+ String[] split = relativeFilename.split("[\\\\/]");
+ String filename = split[split.length - 1];
+ Texture tex = null;
+ try {
+ tex = assetManager.loadTexture(new ContentTextureKey(scene.currentAssetInfo.getKey().getFolder() + filename, content));
+ } catch(Exception e) {}
+ if(tex != null)
+ image = tex.getImage();
+ }
+ }
+ if(image == null)
+ return new Image(Image.Format.RGB8, 1, 1, BufferUtils.createByteBuffer((int) ((long) 1 * (long) 1 * (long) Image.Format.RGB8.getBitsPerPixel() / 8L)), ColorSpace.Linear);
+ return image;
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxMaterial.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxMaterial.java
new file mode 100644
index 000000000..c16d253ad
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxMaterial.java
@@ -0,0 +1,113 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import com.jme3.material.Material;
+import com.jme3.material.RenderState.BlendMode;
+import com.jme3.math.ColorRGBA;
+import com.jme3.math.Vector3f;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxMaterial extends FbxObject {
+
+ public String shadingModel = "phong";
+ public Vector3f ambientColor = new Vector3f(0.2f, 0.2f, 0.2f);
+ public float ambientFactor = 1.0f;
+ public Vector3f diffuseColor = new Vector3f(0.8f, 0.8f, 0.8f);
+ public float diffuseFactor = 1.0f;
+ public Vector3f specularColor = new Vector3f(0.2f, 0.2f, 0.2f);
+ public float specularFactor = 1.0f;
+ public float shininessExponent = 1.0f;
+
+ public Material material;
+
+ public FbxMaterial(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ if(type.equals("")) {
+ for(FbxElement e : element.children) {
+ if(e.id.equals("ShadingModel")) {
+ shadingModel = (String) e.properties.get(0);
+ } else if(e.id.equals("Properties70")) {
+ for(FbxElement e2 : e.children) {
+ if(e2.id.equals("P")) {
+ double x, y, z;
+ String propName = (String) e2.properties.get(0);
+ switch(propName) {
+ case "AmbientColor":
+ x = (Double) e2.properties.get(4);
+ y = (Double) e2.properties.get(5);
+ z = (Double) e2.properties.get(6);
+ ambientColor.set((float) x, (float) y, (float) z);
+ break;
+ case "AmbientFactor":
+ x = (Double) e2.properties.get(4);
+ ambientFactor = (float) x;
+ break;
+ case "DiffuseColor":
+ x = (Double) e2.properties.get(4);
+ y = (Double) e2.properties.get(5);
+ z = (Double) e2.properties.get(6);
+ diffuseColor.set((float) x, (float) y, (float) z);
+ break;
+ case "DiffuseFactor":
+ x = (Double) e2.properties.get(4);
+ diffuseFactor = (float) x;
+ break;
+ case "SpecularColor":
+ x = (Double) e2.properties.get(4);
+ y = (Double) e2.properties.get(5);
+ z = (Double) e2.properties.get(6);
+ specularColor.set((float) x, (float) y, (float) z);
+ break;
+ case "Shininess":
+ case "ShininessExponent":
+ x = (Double) e2.properties.get(4);
+ shininessExponent = (float) x;
+ break;
+ }
+ }
+ }
+ }
+ }
+ material = createMaterial();
+ }
+ }
+
+ @Override
+ public void link(FbxObject otherObject, String propertyName) {
+ if(otherObject instanceof FbxTexture) {
+ FbxTexture tex = (FbxTexture) otherObject;
+ if(tex.texture == null || material == null)
+ return;
+ switch(propertyName) {
+ case "DiffuseColor":
+ material.setTexture("DiffuseMap", tex.texture);
+ material.setColor("Diffuse", ColorRGBA.White);
+ break;
+ case "SpecularColor":
+ material.setTexture("SpecularMap", tex.texture);
+ material.setColor("Specular", ColorRGBA.White);
+ break;
+ case "NormalMap":
+ material.setTexture("NormalMap", tex.texture);
+ break;
+ }
+ }
+ }
+
+ private Material createMaterial() {
+ Material m = new Material(scene.assetManager, "Common/MatDefs/Light/Lighting.j3md");
+ m.setName(name);
+ ambientColor.multLocal(ambientFactor);
+ diffuseColor.multLocal(diffuseFactor);
+ specularColor.multLocal(specularFactor);
+ m.setColor("Ambient", new ColorRGBA(ambientColor.x, ambientColor.y, ambientColor.z, 1));
+ m.setColor("Diffuse", new ColorRGBA(diffuseColor.x, diffuseColor.y, diffuseColor.z, 1));
+ m.setColor("Specular", new ColorRGBA(specularColor.x, specularColor.y, specularColor.z, 1));
+ m.setFloat("Shininess", shininessExponent);
+ m.setBoolean("UseMaterialColors", true);
+ m.setFloat("AlphaDiscardThreshold", 0.5f); // TODO replace with right way in JME to set "Aplha Test"
+ m.getAdditionalRenderState().setBlendMode(BlendMode.Alpha);
+ return m;
+ }
+
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxMesh.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxMesh.java
new file mode 100644
index 000000000..930dd830d
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxMesh.java
@@ -0,0 +1,537 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import java.io.IOException;
+import java.nio.FloatBuffer;
+import java.util.ArrayList;
+import java.util.Iterator;
+import java.util.List;
+
+import com.jme3.asset.AssetLoadException;
+import com.jme3.scene.Geometry;
+import com.jme3.scene.Mesh;
+import com.jme3.scene.VertexBuffer;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+import com.jme3.scene.Mesh.Mode;
+import com.jme3.scene.Node;
+import com.jme3.util.BufferUtils;
+import com.jme3.util.IntMap;
+import com.jme3.util.IntMap.Entry;
+
+public class FbxMesh extends FbxObject {
+
+ public double[] vertices;
+ public int[] indices;
+ public int[] edges;
+ public String normalsMapping;
+ public String normalsReference;
+ public double[] normals;
+ public String tangentsMapping;
+ public String tangentsReference;
+ public double[] tangents;
+ public String binormalsMapping;
+ public String binormalsReference;
+ public double[] binormals;
+ public String uvMapping;
+ public String uvReference;
+ public double[] uv;
+ public int[] uvIndex;
+ public List uvIndexes = new ArrayList<>();
+ public List uvs = new ArrayList<>();
+ public String smoothingMapping;
+ public String smoothingReference;
+ public int[] smoothing;
+ public String materialsMapping;
+ public String materialsReference;
+ public int[] materials;
+ // Build helping data
+ public int iCount;
+ public int vCount;
+ public int srcVertexCount;
+ public List vertexMap; // Target vertex -> source vertex
+ public List> reverseVertexMap; // source vertex -> list of target vertices
+ public List indexMap; // Target vertex -> source index
+
+ public List geometries; // One mesh can be split in two geometries in case of by-polygon material mapping
+ public FbxNode parent;
+ public int lastMaterialId = 0;
+
+ public FbxMesh(SceneLoader scene, FbxElement element) throws IOException {
+ super(scene, element);
+ if(type.equals("Mesh")) {
+ data: for(FbxElement e : element.children) {
+ switch(e.id) {
+ case "Vertices":
+ vertices = (double[]) e.properties.get(0);
+ break;
+ case "PolygonVertexIndex":
+ indices = (int[]) e.properties.get(0);
+ break;
+ // TODO edges are not used now
+ /*case "Edges":
+ edges = (int[]) e.properties.get(0);
+ break;*/
+ case "LayerElementNormal":
+ for(FbxElement e2 : e.children) {
+ switch(e2.id) {
+ case "MappingInformationType":
+ normalsMapping = (String) e2.properties.get(0);
+ if(!normalsMapping.equals("ByVertice") && !normalsMapping.equals("ByPolygonVertex")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementNormal.MappingInformationType attribute (" + normalsReference + ")");
+ continue data;
+ }
+ break;
+ case "ReferenceInformationType":
+ normalsReference = (String) e2.properties.get(0);
+ if(!normalsReference.equals("Direct")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementNormal.ReferenceInformationType attribute (" + normalsReference + ")");
+ continue data;
+ }
+ break;
+ case "Normals":
+ normals = (double[]) e2.properties.get(0);
+ break;
+ }
+ }
+ break;
+ case "LayerElementTangent":
+ for(FbxElement e2 : e.children) {
+ switch(e2.id) {
+ case "MappingInformationType":
+ tangentsMapping = (String) e2.properties.get(0);
+ if(!tangentsMapping.equals("ByVertice") && !tangentsMapping.equals("ByPolygonVertex")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementTangent.MappingInformationType attribute (" + tangentsMapping + ")");
+ continue data;
+ }
+ break;
+ case "ReferenceInformationType":
+ tangentsReference = (String) e2.properties.get(0);
+ if(!tangentsReference.equals("Direct")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementTangent.ReferenceInformationType attribute (" + tangentsReference + ")");
+ continue data;
+ }
+ break;
+ case "Tangents":
+ tangents = (double[]) e2.properties.get(0);
+ break;
+ }
+ }
+ break;
+ case "LayerElementBinormal":
+ for(FbxElement e2 : e.children) {
+ switch(e2.id) {
+ case "MappingInformationType":
+ binormalsMapping = (String) e2.properties.get(0);
+ if(!binormalsMapping.equals("ByVertice") && !binormalsMapping.equals("ByPolygonVertex")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementBinormal.MappingInformationType attribute (" + binormalsMapping + ")");
+ continue data;
+ }
+ break;
+ case "ReferenceInformationType":
+ binormalsReference = (String) e2.properties.get(0);
+ if(!binormalsReference.equals("Direct")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementBinormal.ReferenceInformationType attribute (" + binormalsReference + ")");
+ continue data;
+ }
+ break;
+ case "Tangents":
+ binormals = (double[]) e2.properties.get(0);
+ break;
+ }
+ }
+ break;
+ case "LayerElementUV":
+ for(FbxElement e2 : e.children) {
+ switch(e2.id) {
+ case "MappingInformationType":
+ uvMapping = (String) e2.properties.get(0);
+ if(!uvMapping.equals("ByPolygonVertex")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementUV.MappingInformationType attribute (" + uvMapping + ")");
+ continue data;
+ }
+ break;
+ case "ReferenceInformationType":
+ uvReference = (String) e2.properties.get(0);
+ if(!uvReference.equals("IndexToDirect")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementUV.ReferenceInformationType attribute (" + uvReference + ")");
+ continue data;
+ }
+ break;
+ case "UV":
+ uv = (double[]) e2.properties.get(0);
+ uvs.add(uv);
+ break;
+ case "UVIndex":
+ uvIndex = (int[]) e2.properties.get(0);
+ uvIndexes.add(uvIndex);
+ break;
+ }
+ }
+ break;
+ // TODO smoothing is not used now
+ /*case "LayerElementSmoothing":
+ for(FBXElement e2 : e.children) {
+ switch(e2.id) {
+ case "MappingInformationType":
+ smoothingMapping = (String) e2.properties.get(0);
+ if(!smoothingMapping.equals("ByEdge"))
+ throw new AssetLoadException("Not supported LayerElementSmoothing.MappingInformationType = " + smoothingMapping);
+ break;
+ case "ReferenceInformationType":
+ smoothingReference = (String) e2.properties.get(0);
+ if(!smoothingReference.equals("Direct"))
+ throw new AssetLoadException("Not supported LayerElementSmoothing.ReferenceInformationType = " + smoothingReference);
+ break;
+ case "Smoothing":
+ smoothing = (int[]) e2.properties.get(0);
+ break;
+ }
+ }
+ break;*/
+ case "LayerElementMaterial":
+ for(FbxElement e2 : e.children) {
+ switch(e2.id) {
+ case "MappingInformationType":
+ materialsMapping = (String) e2.properties.get(0);
+ if(!materialsMapping.equals("AllSame") && !materialsMapping.equals("ByPolygon")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementMaterial.MappingInformationType attribute (" + materialsMapping + ")");
+ continue data;
+ }
+ break;
+ case "ReferenceInformationType":
+ materialsReference = (String) e2.properties.get(0);
+ if(!materialsReference.equals("IndexToDirect")) {
+ if(SceneLoader.WARN_IGNORED_ATTRIBUTES)
+ scene.warning("Ignored LayerElementMaterial.ReferenceInformationType attribute (" + materialsReference + ")");
+ continue data;
+ }
+ break;
+ case "Materials":
+ materials = (int[]) e2.properties.get(0);
+ break;
+ }
+ }
+ break;
+ }
+ }
+ geometries = createGeometries();
+ }
+ }
+
+ public void setParent(Node node) {
+ for(int i = 0; i < geometries.size(); ++i) {
+ Geometry geom = geometries.get(i);
+ geom.setName(node.getName() + (i > 0 ? "-" + i : ""));
+ geom.updateModelBound();
+ node.attachChild(geom);
+ }
+ }
+
+ @Override
+ public void linkToZero() {
+ setParent(scene.sceneNode);
+ }
+
+ public void clearMaterials() {
+ for(Geometry g : geometries) {
+ if(g.getUserData("FBXMaterial") != null)
+ g.setUserData("FBXMaterial", null);
+ }
+ }
+
+ @Override
+ public void link(FbxObject otherObject) {
+ if(otherObject instanceof FbxSkin) {
+ FbxSkin skin = (FbxSkin) otherObject;
+ skin.toSkin.add(this);
+ }
+ }
+
+ private List createGeometries() throws IOException {
+ Mesh mesh = new Mesh();
+ mesh.setMode(Mode.Triangles);
+ // Since each vertex should contain unique texcoord and normal we should unroll vertex indexing
+ // So we don't use VertexBuffer.Type.Index for elements drawing
+ // Moreover quads should be triangulated (this increases number of vertices)
+ if(indices != null) {
+ iCount = indices.length;
+ srcVertexCount = vertices.length / 3;
+ // Indices contains negative numbers to define polygon last index
+ // Check indices strides to be sure we have triangles or quads
+ vCount = 0;
+ // Count number of vertices to be produced
+ int polyVertCount = 0;
+ for(int i = 0; i < iCount; ++i) {
+ int index = indices[i];
+ polyVertCount++;
+ if(index < 0) {
+ if(polyVertCount == 3) {
+ vCount += 3; // A triangle
+ } else if(polyVertCount == 4) {
+ vCount += 6; // A quad produce two triangles
+ } else {
+ throw new AssetLoadException("Unsupported PolygonVertexIndex stride");
+ }
+ polyVertCount = 0;
+ }
+ }
+ // Unroll index array into vertex mapping
+ vertexMap = new ArrayList<>(vCount);
+ indexMap = new ArrayList<>(vCount);
+ polyVertCount = 0;
+ for(int i = 0; i < iCount; ++i) {
+ int index = indices[i];
+ polyVertCount++;
+ if(index < 0) {
+ int lastIndex = -(index + 1);
+ if(polyVertCount == 3) {
+ vertexMap.add(indices[i - 2]);
+ vertexMap.add(indices[i - 1]);
+ vertexMap.add(lastIndex);
+ indexMap.add(i - 2);
+ indexMap.add(i - 1);
+ indexMap.add(i - 0);
+ } else if(polyVertCount == 4) {
+ vertexMap.add(indices[i - 3]);
+ vertexMap.add(indices[i - 2]);
+ vertexMap.add(indices[i - 1]);
+ vertexMap.add(indices[i - 3]);
+ vertexMap.add(indices[i - 1]);
+ vertexMap.add(lastIndex);
+ indexMap.add(i - 3);
+ indexMap.add(i - 2);
+ indexMap.add(i - 1);
+ indexMap.add(i - 3);
+ indexMap.add(i - 1);
+ indexMap.add(i - 0);
+ }
+ polyVertCount = 0;
+ }
+ }
+ // Build reverse vertex mapping
+ reverseVertexMap = new ArrayList<>(srcVertexCount);
+ for(int i = 0; i < srcVertexCount; ++i)
+ reverseVertexMap.add(new ArrayList());
+ for(int i = 0; i < vCount; ++i) {
+ int index = vertexMap.get(i);
+ reverseVertexMap.get(index).add(i);
+ }
+ } else {
+ // Stub for no vertex indexing (direct mapping)
+ iCount = vCount = srcVertexCount;
+ vertexMap = new ArrayList<>(vCount);
+ indexMap = new ArrayList<>(vCount);
+ reverseVertexMap = new ArrayList<>(vCount);
+ for(int i = 0; i < vCount; ++i) {
+ vertexMap.set(i, i);
+ indexMap.set(i, i);
+ List l = new ArrayList(1);
+ l.add(i);
+ reverseVertexMap.add(l);
+ }
+ }
+ if(vertices != null) {
+ // Unroll vertices data array
+ FloatBuffer posBuf = BufferUtils.createFloatBuffer(vCount * 3);
+ mesh.setBuffer(VertexBuffer.Type.Position, 3, posBuf);
+ int srcCount = vertices.length / 3;
+ for(int i = 0; i < vCount; ++i) {
+ int index = vertexMap.get(i);
+ if(index > srcCount)
+ throw new AssetLoadException("Invalid vertex mapping. Unexpected lookup vertex " + index + " from " + srcCount);
+ float x = (float) vertices[3 * index + 0] / scene.unitSize * scene.xAxis; // XXX Why we should scale by unit size?
+ float y = (float) vertices[3 * index + 1] / scene.unitSize * scene.yAxis;
+ float z = (float) vertices[3 * index + 2] / scene.unitSize * scene.zAxis;
+ posBuf.put(x).put(y).put(z);
+ }
+ }
+ if(normals != null) {
+ // Unroll normals data array
+ FloatBuffer normBuf = BufferUtils.createFloatBuffer(vCount * 3);
+ mesh.setBuffer(VertexBuffer.Type.Normal, 3, normBuf);
+ List mapping = null;
+ if(normalsMapping.equals("ByVertice"))
+ mapping = vertexMap;
+ else if(normalsMapping.equals("ByPolygonVertex"))
+ mapping = indexMap;
+ else
+ throw new IOException("Unknown normals mapping type: " + normalsMapping);
+ int srcCount = normals.length / 3;
+ for(int i = 0; i < vCount; ++i) {
+ int index = mapping.get(i);
+ if(index > srcCount)
+ throw new AssetLoadException("Invalid normal mapping. Unexpected lookup normal " + index + " from " + srcCount);
+ float x = (float) normals[3 * index + 0] * scene.xAxis;
+ float y = (float) normals[3 * index + 1] * scene.yAxis;
+ float z = (float) normals[3 * index + 2] * scene.zAxis;
+ normBuf.put(x).put(y).put(z);
+ }
+ }
+ if(tangents != null) {
+ // Unroll normals data array
+ FloatBuffer tanBuf = BufferUtils.createFloatBuffer(vCount * 4);
+ mesh.setBuffer(VertexBuffer.Type.Tangent, 4, tanBuf);
+ List mapping = null;
+ if(tangentsMapping.equals("ByVertice"))
+ mapping = vertexMap;
+ else if(tangentsMapping.equals("ByPolygonVertex"))
+ mapping = indexMap;
+ else
+ throw new IOException("Unknown tangents mapping type: " + tangentsMapping);
+ int srcCount = tangents.length / 3;
+ for(int i = 0; i < vCount; ++i) {
+ int index = mapping.get(i);
+ if(index > srcCount)
+ throw new AssetLoadException("Invalid tangent mapping. Unexpected lookup tangent " + index + " from " + srcCount);
+ float x = (float) tangents[3 * index + 0] * scene.xAxis;
+ float y = (float) tangents[3 * index + 1] * scene.yAxis;
+ float z = (float) tangents[3 * index + 2] * scene.zAxis;
+ tanBuf.put(x).put(y).put(z).put(-1.0f);
+ }
+ }
+ if(binormals != null) {
+ // Unroll normals data array
+ FloatBuffer binormBuf = BufferUtils.createFloatBuffer(vCount * 3);
+ mesh.setBuffer(VertexBuffer.Type.Binormal, 3, binormBuf);
+ List mapping = null;
+ if(binormalsMapping.equals("ByVertice"))
+ mapping = vertexMap;
+ else if(binormalsMapping.equals("ByPolygonVertex"))
+ mapping = indexMap;
+ else
+ throw new IOException("Unknown binormals mapping type: " + binormalsMapping);
+ int srcCount = binormals.length / 3;
+ for(int i = 0; i < vCount; ++i) {
+ int index = mapping.get(i);
+ if(index > srcCount)
+ throw new AssetLoadException("Invalid binormal mapping. Unexpected lookup binormal " + index + " from " + srcCount);
+ float x = (float) binormals[3 * index + 0] * scene.xAxis;
+ float y = (float) binormals[3 * index + 1] * scene.yAxis;
+ float z = (float) binormals[3 * index + 2] * scene.zAxis;
+ binormBuf.put(x).put(y).put(z);
+ }
+ }
+ for(int uvLayer = 0; uvLayer < uvs.size(); ++uvLayer) {
+ double[] uv = uvs.get(uvLayer);
+ int[] uvIndex = uvIndexes.size() > uvLayer ? uvIndexes.get(uvLayer) : null;
+ List unIndexMap = vertexMap;
+ if(uvIndex != null) {
+ int uvIndexSrcCount = uvIndex.length;
+ if(uvIndexSrcCount != iCount)
+ throw new AssetLoadException("Invalid number of texcoord index data " + uvIndexSrcCount + " expected " + iCount);
+ // Unroll UV index array
+ unIndexMap = new ArrayList<>(vCount);
+ int polyVertCount = 0;
+ for(int i = 0; i < iCount; ++i) {
+ int index = indices[i];
+ polyVertCount++;
+ if(index < 0) {
+ if(polyVertCount == 3) {
+ unIndexMap.add(uvIndex[i - 2]);
+ unIndexMap.add(uvIndex[i - 1]);
+ unIndexMap.add(uvIndex[i - 0]);
+ } else if(polyVertCount == 4) {
+ unIndexMap.add(uvIndex[i - 3]);
+ unIndexMap.add(uvIndex[i - 2]);
+ unIndexMap.add(uvIndex[i - 1]);
+ unIndexMap.add(uvIndex[i - 3]);
+ unIndexMap.add(uvIndex[i - 1]);
+ unIndexMap.add(uvIndex[i - 0]);
+ }
+ polyVertCount = 0;
+ }
+ }
+ }
+ // Unroll UV data array
+ FloatBuffer tcBuf = BufferUtils.createFloatBuffer(vCount * 2);
+ VertexBuffer.Type type = VertexBuffer.Type.TexCoord;
+ switch(uvLayer) {
+ case 1:
+ type = VertexBuffer.Type.TexCoord2;
+ break;
+ case 2:
+ type = VertexBuffer.Type.TexCoord3;
+ break;
+ case 3:
+ type = VertexBuffer.Type.TexCoord4;
+ break;
+ case 4:
+ type = VertexBuffer.Type.TexCoord5;
+ break;
+ case 5:
+ type = VertexBuffer.Type.TexCoord6;
+ break;
+ case 6:
+ type = VertexBuffer.Type.TexCoord7;
+ break;
+ case 7:
+ type = VertexBuffer.Type.TexCoord8;
+ break;
+ }
+ mesh.setBuffer(type, 2, tcBuf);
+ int srcCount = uv.length / 2;
+ for(int i = 0; i < vCount; ++i) {
+ int index = unIndexMap.get(i);
+ if(index > srcCount)
+ throw new AssetLoadException("Invalid texcoord mapping. Unexpected lookup texcoord " + index + " from " + srcCount);
+ float u = (index >= 0) ? (float) uv[2 * index + 0] : 0;
+ float v = (index >= 0) ? (float) uv[2 * index + 1] : 0;
+ tcBuf.put(u).put(v);
+ }
+ }
+ List geometries = new ArrayList();
+ if(materialsReference.equals("IndexToDirect") && materialsMapping.equals("ByPolygon")) {
+ IntMap> indexBuffers = new IntMap<>();
+ for(int polygon = 0; polygon < materials.length; ++polygon) {
+ int material = materials[polygon];
+ List list = indexBuffers.get(material);
+ if(list == null) {
+ list = new ArrayList<>();
+ indexBuffers.put(material, list);
+ }
+ list.add(polygon * 3 + 0);
+ list.add(polygon * 3 + 1);
+ list.add(polygon * 3 + 2);
+ }
+ Iterator>> iterator = indexBuffers.iterator();
+ while(iterator.hasNext()) {
+ Entry> e = iterator.next();
+ int materialId = e.getKey();
+ List indexes = e.getValue();
+ Mesh newMesh = mesh.clone();
+ newMesh.setBuffer(VertexBuffer.Type.Index, 3, toArray(indexes.toArray(new Integer[indexes.size()])));
+ newMesh.setStatic();
+ newMesh.updateBound();
+ newMesh.updateCounts();
+ Geometry geom = new Geometry();
+ geom.setMesh(newMesh);
+ geometries.add(geom);
+ geom.setUserData("FBXMaterial", materialId);
+ }
+ } else {
+ mesh.setStatic();
+ mesh.updateBound();
+ mesh.updateCounts();
+ Geometry geom = new Geometry();
+ geom.setMesh(mesh);
+ geometries.add(geom);
+ }
+ return geometries;
+ }
+
+ private static int[] toArray(Integer[] arr) {
+ int[] ret = new int[arr.length];
+ for(int i = 0; i < arr.length; ++i)
+ ret[i] = arr[i].intValue();
+ return ret;
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxNode.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxNode.java
new file mode 100644
index 000000000..ebb209492
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxNode.java
@@ -0,0 +1,272 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import java.util.HashMap;
+import java.util.Map;
+
+import com.jme3.animation.Bone;
+import com.jme3.animation.Skeleton;
+import com.jme3.material.Material;
+import com.jme3.material.RenderState.FaceCullMode;
+import com.jme3.math.Matrix4f;
+import com.jme3.math.Quaternion;
+import com.jme3.math.Transform;
+import com.jme3.math.Vector3f;
+import com.jme3.scene.Geometry;
+import com.jme3.scene.Node;
+import com.jme3.scene.plugins.fbx.RotationOrder;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxNode extends FbxObject {
+
+ public FaceCullMode cullMode = FaceCullMode.Back;
+ public Transform localTransform;
+ public Node node;
+ public FbxNode parentFbxNode;
+
+ public boolean rotationActive = false;
+ public RotationOrder rotationOrder = RotationOrder.EULER_XYZ;
+
+
+ // For bones and animation, in world space
+ public Matrix4f bindTransform = null;
+ public int boneIndex;
+ public Map animTranslations = new HashMap<>();
+ public Map animRotations = new HashMap<>();
+ public Map animScales = new HashMap<>();
+ public Bone bone;
+ private FbxAnimNode lastAnimTranslation;
+ private FbxAnimNode lastAnimRotation;
+ private FbxAnimNode lastAnimScale;
+ private FbxMesh mesh;
+ public Map skinToCluster = new HashMap<>();
+
+ public FbxNode(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ node = new Node(name);
+ Vector3f translationLocalRaw = new Vector3f();
+ Vector3f rotationOffsetRaw = new Vector3f();
+ Vector3f rotationPivotRaw = new Vector3f();
+ Vector3f rotationPreRaw = new Vector3f();
+ Vector3f rotationLocalRaw = new Vector3f();
+ Vector3f rotationPostRaw = new Vector3f();
+ Vector3f scaleOffsetRaw = new Vector3f();
+ Vector3f scalePivotRaw = new Vector3f();
+ Vector3f scaleLocalRaw = new Vector3f(1, 1, 1);
+ for(FbxElement prop : element.getFbxProperties()) {
+ double x, y, z;
+ String propName = (String) prop.properties.get(0);
+ switch(propName) {
+ case "RotationOrder":
+ rotationOrder = RotationOrder.values[(Integer) prop.properties.get(4)];
+ break;
+ case "Lcl Translation":
+ readVectorFromProp(translationLocalRaw, prop);
+ break;
+ case "Lcl Rotation":
+ readVectorFromProp(rotationLocalRaw, prop);
+ break;
+ case "Lcl Scaling":
+ readVectorFromProp(scaleLocalRaw, prop);
+ break;
+ case "PreRotation":
+ readVectorFromProp(rotationPreRaw, prop);
+ break;
+ case "RotationActive":
+ rotationActive = ((Number) prop.properties.get(4)).intValue() == 1;
+ break;
+ case "RotationPivot":
+ readVectorFromProp(rotationPivotRaw, prop);
+ break;
+ case "PostRotation":
+ readVectorFromProp(rotationPostRaw, prop);
+ break;
+ case "ScaleOffset":
+ readVectorFromProp(scaleOffsetRaw, prop);
+ break;
+ case "ScalePivot":
+ readVectorFromProp(scalePivotRaw, prop);
+ break;
+ case "U":
+ String userDataKey = (String) prop.properties.get(0);
+ String userDataType = (String) prop.properties.get(1);
+ Object userDataValue;
+ if(userDataType.equals("KString")) {
+ userDataValue = (String) prop.properties.get(4);
+ } else if(userDataType.equals("int")) {
+ userDataValue = (Integer) prop.properties.get(4);
+ } else if(userDataType.equals("double")) {
+ // NOTE: jME3 does not support doubles in UserData.
+ // Need to convert to float.
+ userDataValue = ((Double) prop.properties.get(4)).floatValue();
+ } else if(userDataType.equals("Vector")) {
+ x = (Double) prop.properties.get(4);
+ y = (Double) prop.properties.get(5);
+ z = (Double) prop.properties.get(6);
+ userDataValue = new Vector3f((float) x, (float) y, (float) z);
+ } else {
+ scene.warning("Unsupported user data type: " + userDataType + ". Ignoring.");
+ continue;
+ }
+ node.setUserData(userDataKey, userDataValue);
+ break;
+ }
+ }
+
+ FbxElement cullingElement = element.getChildById("Culling");
+ if(cullingElement != null && cullingElement.properties.get(0).equals("CullingOff"))
+ cullMode = FaceCullMode.Off; // TODO Add other variants
+
+ /*From http://area.autodesk.com/forum/autodesk-fbx/fbx-sdk/the-makeup-of-the-local-matrix-of-an-kfbxnode/
+
+ Local Matrix = LclTranslation * RotationOffset * RotationPivot *
+ PreRotation * LclRotation * PostRotation * RotationPivotInverse *
+ ScalingOffset * ScalingPivot * LclScaling * ScalingPivotInverse
+
+ LocalTranslation : translate (xform -query -translation)
+ RotationOffset: translation compensates for the change in the rotate pivot point (xform -q -rotateTranslation)
+ RotationPivot: current rotate pivot position (xform -q -rotatePivot)
+ PreRotation : joint orientation(pre rotation)
+ LocalRotation: rotate transform (xform -q -rotation & xform -q -rotateOrder)
+ PostRotation : rotate axis (xform -q -rotateAxis)
+ RotationPivotInverse: inverse of RotationPivot
+ ScalingOffset: translation compensates for the change in the scale pivot point (xform -q -scaleTranslation)
+ ScalingPivot: current scale pivot position (xform -q -scalePivot)
+ LocalScaling: scale transform (xform -q -scale)
+ ScalingPivotInverse: inverse of ScalingPivot
+ */
+
+ RotationOrder rotOrder = rotationActive ? rotationOrder : RotationOrder.EULER_XYZ;
+
+ Matrix4f transformMatrix = new Matrix4f();
+ transformMatrix.setTranslation(translationLocalRaw.x + rotationOffsetRaw.x + rotationPivotRaw.x, translationLocalRaw.y + rotationOffsetRaw.y + rotationPivotRaw.y, translationLocalRaw.z + rotationOffsetRaw.z + rotationPivotRaw.z);
+
+ if(rotationActive) {
+ Quaternion postRotation = rotOrder.rotate(rotationPostRaw.x, rotationPostRaw.y, rotationPostRaw.z);
+ Quaternion localRotation = rotOrder.rotate(rotationLocalRaw.x, rotationLocalRaw.y, rotationLocalRaw.z);
+ Quaternion preRotation = rotOrder.rotate(rotationPreRaw.x, rotationPreRaw.y, rotationPreRaw.z);
+ //preRotation.multLocal(localRotation).multLocal(postRotation);
+ postRotation.multLocal(localRotation).multLocal(preRotation);
+ transformMatrix.multLocal(postRotation);
+ } else {
+ transformMatrix.multLocal(rotOrder.rotate(rotationLocalRaw.x, rotationLocalRaw.y, rotationLocalRaw.z));
+ }
+
+ Matrix4f mat = new Matrix4f();
+ mat.setTranslation(scaleOffsetRaw.x + scalePivotRaw.x - rotationPivotRaw.x, scaleOffsetRaw.y + scalePivotRaw.y - rotationPivotRaw.y, scaleOffsetRaw.z + scalePivotRaw.z - rotationPivotRaw.z);
+ transformMatrix.multLocal(mat);
+
+ transformMatrix.scale(scaleLocalRaw);
+ transformMatrix.scale(new Vector3f(scene.unitSize, scene.unitSize, scene.unitSize));
+
+ mat.setTranslation(scalePivotRaw.negate());
+ transformMatrix.multLocal(mat);
+
+ localTransform = new Transform(transformMatrix.toTranslationVector(), transformMatrix.toRotationQuat(), transformMatrix.toScaleVector());
+
+ node.setLocalTransform(localTransform);
+ }
+
+ @Override
+ public void linkToZero() {
+ scene.sceneNode.attachChild(node);
+ }
+
+ public void setSkeleton(Skeleton skeleton) {
+ if(bone != null)
+ boneIndex = skeleton.getBoneIndex(bone);
+ }
+
+ public void buildBindPoseBoneTransform() {
+ if(bone != null) {
+ Matrix4f t = bindTransform;
+ if(t != null) {
+ Matrix4f parentMatrix = parentFbxNode != null ? parentFbxNode.bindTransform : Matrix4f.IDENTITY;
+ if(parentMatrix == null)
+ parentMatrix = node.getLocalToWorldMatrix(null);
+ t = parentMatrix.invert().multLocal(t);
+ bone.setBindTransforms(t.toTranslationVector(), t.toRotationQuat(), t.toScaleVector());
+ } else {
+ bone.setBindTransforms(node.getLocalTranslation(), node.getLocalRotation(), node.getLocalScale());
+ }
+ }
+ }
+
+ @Override
+ public void link(FbxObject child, String propertyName) {
+ if(child instanceof FbxAnimNode) {
+ FbxAnimNode anim = (FbxAnimNode) child;
+ switch(propertyName) {
+ case "Lcl Translation":
+ animTranslations.put(anim.layerId, anim);
+ lastAnimTranslation = anim;
+ break;
+ case "Lcl Rotation":
+ animRotations.put(anim.layerId, anim);
+ lastAnimRotation = anim;
+ break;
+ case "Lcl Scaling":
+ animScales.put(anim.layerId, anim);
+ lastAnimScale = anim;
+ break;
+ }
+ }
+ }
+
+ public FbxAnimNode animTranslation(long layerId) {
+ if(layerId == 0)
+ return lastAnimTranslation;
+ return animTranslations.get(layerId);
+ }
+
+ public FbxAnimNode animRotation(long layerId) {
+ if(layerId == 0)
+ return lastAnimRotation;
+ return animRotations.get(layerId);
+ }
+
+ public FbxAnimNode animScale(long layerId) {
+ if(layerId == 0)
+ return lastAnimScale;
+ return animScales.get(layerId);
+ }
+
+ @Override
+ public void link(FbxObject otherObject) {
+ if(otherObject instanceof FbxMaterial) {
+ FbxMaterial m = (FbxMaterial) otherObject;
+ Material mat = m.material;
+ if(cullMode != FaceCullMode.Back)
+ mat.getAdditionalRenderState().setFaceCullMode(cullMode);
+ for(Geometry g : mesh.geometries) {
+ if(g.getUserData("FBXMaterial") != null) {
+ if((Integer) g.getUserData("FBXMaterial") == mesh.lastMaterialId)
+ g.setMaterial(mat);
+ } else {
+ g.setMaterial(mat);
+ }
+ }
+ mesh.lastMaterialId++;
+ } else if(otherObject instanceof FbxNode) {
+ FbxNode n = (FbxNode) otherObject;
+ node.attachChild(n.node);
+ n.parentFbxNode = this;
+ if(isLimb() && n.isLimb()) {
+ if(bone == null)
+ bone = new Bone(name);
+ if(n.bone == null)
+ n.bone = new Bone(n.name);
+ bone.addChild(n.bone);
+ }
+ } else if(otherObject instanceof FbxMesh) {
+ FbxMesh m = (FbxMesh) otherObject;
+ m.setParent(node);
+ m.parent = this;
+ mesh = m;
+ }
+ }
+
+ public boolean isLimb() {
+ return type.equals("LimbNode");
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxObject.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxObject.java
new file mode 100644
index 000000000..57611f072
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxObject.java
@@ -0,0 +1,40 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import com.jme3.math.Vector3f;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxObject {
+
+ protected final SceneLoader scene;
+ public final FbxElement element;
+ public final long id;
+ public final String name;
+ public final String type;
+
+ public FbxObject(SceneLoader scene, FbxElement element) {
+ this.scene = scene;
+ this.element = element;
+ this.id = (Long) element.properties.get(0);
+ String name = (String) element.properties.get(1);
+ this.name = name.substring(0, name.indexOf(0));
+ this.type = (String) element.properties.get(2);
+ }
+
+ public void link(FbxObject child) {
+ }
+
+ public void link(FbxObject child, String propertyName) {
+ }
+
+ // Parent is 0 id
+ public void linkToZero() {
+ }
+
+ protected static void readVectorFromProp(Vector3f store, FbxElement propElement) {
+ float x = ((Double) propElement.properties.get(4)).floatValue();
+ float y = ((Double) propElement.properties.get(5)).floatValue();
+ float z = ((Double) propElement.properties.get(6)).floatValue();
+ store.set(x, y, z);
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxSkin.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxSkin.java
new file mode 100644
index 000000000..896860077
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxSkin.java
@@ -0,0 +1,150 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import java.nio.ByteBuffer;
+import java.nio.FloatBuffer;
+import java.util.ArrayList;
+import java.util.List;
+
+import com.jme3.asset.AssetLoadException;
+import com.jme3.scene.Mesh;
+import com.jme3.scene.VertexBuffer;
+import com.jme3.scene.VertexBuffer.Type;
+import com.jme3.scene.VertexBuffer.Usage;
+import com.jme3.util.BufferUtils;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxSkin extends FbxObject {
+
+ public String skinningType;
+ public List toSkin = new ArrayList<>();
+ public List bones = new ArrayList<>();
+
+ public FbxSkin(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ for(FbxElement e : element.children) {
+ switch(e.id) {
+ case "SkinningType":
+ skinningType = (String) e.properties.get(0);
+ break;
+ }
+ }
+ }
+
+ @Override
+ public void link(FbxObject otherObject) {
+ if(otherObject instanceof FbxCluster) {
+ FbxCluster cluster = ((FbxCluster) otherObject);
+ cluster.skin = this;
+ }
+ }
+
+ public void generateSkinning() {
+ for(FbxMesh fbxMesh : toSkin) {
+ if(fbxMesh.geometries == null)
+ continue;
+ Mesh firstMesh = fbxMesh.geometries.get(0).getMesh();
+ int maxWeightsPerVert = generateBoneData(firstMesh, fbxMesh);
+ for(int i = 0; i < fbxMesh.geometries.size(); ++i) {
+ Mesh mesh = fbxMesh.geometries.get(i).getMesh();
+ if(mesh != firstMesh) {
+ mesh.setBuffer(firstMesh.getBuffer(VertexBuffer.Type.BoneWeight));
+ mesh.setBuffer(firstMesh.getBuffer(VertexBuffer.Type.BoneIndex));
+ mesh.setBuffer(firstMesh.getBuffer(VertexBuffer.Type.HWBoneWeight));
+ mesh.setBuffer(firstMesh.getBuffer(VertexBuffer.Type.HWBoneIndex));
+ }
+ mesh.setMaxNumWeights(maxWeightsPerVert);
+ mesh.generateBindPose(true);
+ }
+ }
+ }
+
+ private int generateBoneData(Mesh mesh, FbxMesh fbxMesh) {
+ // Create bone buffers
+ FloatBuffer boneWeightData = BufferUtils.createFloatBuffer(fbxMesh.vCount * 4);
+ ByteBuffer boneIndicesData = BufferUtils.createByteBuffer(fbxMesh.vCount * 4);
+ mesh.setBuffer(VertexBuffer.Type.BoneWeight, 4, boneWeightData);
+ mesh.setBuffer(VertexBuffer.Type.BoneIndex, 4, boneIndicesData);
+ mesh.getBuffer(VertexBuffer.Type.BoneWeight).setUsage(Usage.CpuOnly);
+ mesh.getBuffer(VertexBuffer.Type.BoneIndex).setUsage(Usage.CpuOnly);
+ VertexBuffer weightsHW = new VertexBuffer(Type.HWBoneWeight);
+ VertexBuffer indicesHW = new VertexBuffer(Type.HWBoneIndex);
+ indicesHW.setUsage(Usage.CpuOnly); // Setting usage to CpuOnly so that the buffer is not send empty to the GPU
+ weightsHW.setUsage(Usage.CpuOnly);
+ mesh.setBuffer(weightsHW);
+ mesh.setBuffer(indicesHW);
+ int bonesLimitExceeded = 0;
+ // Accumulate skin bones influence into mesh buffers
+ for(FbxNode limb : bones) {
+ FbxCluster cluster = limb.skinToCluster.get(id);
+ if(cluster == null || cluster.indexes == null || cluster.weights == null || cluster.indexes.length != cluster.weights.length)
+ continue;
+ if(limb.boneIndex > 255)
+ throw new AssetLoadException("Bone index can't be packed into byte");
+ for(int i = 0; i < cluster.indexes.length; ++i) {
+ int vertexIndex = cluster.indexes[i];
+ if(vertexIndex >= fbxMesh.reverseVertexMap.size())
+ throw new AssetLoadException("Invalid skinning vertex index. Unexpected index lookup " + vertexIndex + " from " + fbxMesh.reverseVertexMap.size());
+ List dstVertices = fbxMesh.reverseVertexMap.get(vertexIndex);
+ for(int j = 0; j < dstVertices.size(); ++j) {
+ int v = dstVertices.get(j);
+ // Append bone index and weight to vertex
+ int offset;
+ int smalestOffset = 0;
+ float w = 0;
+ float smalestW = Float.MAX_VALUE;
+ for(offset = v * 4; offset < v * 4 + 4; ++offset) {
+ w = boneWeightData.get(offset);
+ if(w == 0)
+ break;
+ if(w < smalestW) {
+ smalestW = w;
+ smalestOffset = offset;
+ }
+ }
+ if(w == 0) {
+ boneWeightData.put(offset, (float) cluster.weights[i]);
+ boneIndicesData.put(offset, (byte) limb.boneIndex);
+ } else {
+ if((float) cluster.weights[i] > smalestW) { // If current weight more than smallest, discard smallest
+ boneWeightData.put(smalestOffset, (float) cluster.weights[i]);
+ boneIndicesData.put(smalestOffset, (byte) limb.boneIndex);
+ }
+ bonesLimitExceeded++;
+ }
+ }
+ }
+ }
+ if(bonesLimitExceeded > 0)
+ scene.warning("Skinning support max 4 bone per vertex. Exceeding data of " + bonesLimitExceeded + " weights in mesh bones will be discarded");
+ // Postprocess bones weights
+ int maxWeightsPerVert = 0;
+ boneWeightData.rewind();
+ for(int v = 0; v < fbxMesh.vCount; v++) {
+ float w0 = boneWeightData.get();
+ float w1 = boneWeightData.get();
+ float w2 = boneWeightData.get();
+ float w3 = boneWeightData.get();
+ if(w3 != 0) {
+ maxWeightsPerVert = Math.max(maxWeightsPerVert, 4);
+ } else if(w2 != 0) {
+ maxWeightsPerVert = Math.max(maxWeightsPerVert, 3);
+ } else if(w1 != 0) {
+ maxWeightsPerVert = Math.max(maxWeightsPerVert, 2);
+ } else if(w0 != 0) {
+ maxWeightsPerVert = Math.max(maxWeightsPerVert, 1);
+ }
+ float sum = w0 + w1 + w2 + w3;
+ if(sum != 1f) {
+ // normalize weights
+ float mult = (sum != 0) ? (1f / sum) : 0;
+ boneWeightData.position(v * 4);
+ boneWeightData.put(w0 * mult);
+ boneWeightData.put(w1 * mult);
+ boneWeightData.put(w2 * mult);
+ boneWeightData.put(w3 * mult);
+ }
+ }
+ return maxWeightsPerVert;
+ }
+}
diff --git a/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxTexture.java b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxTexture.java
new file mode 100644
index 000000000..6093d7228
--- /dev/null
+++ b/jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/objects/FbxTexture.java
@@ -0,0 +1,42 @@
+package com.jme3.scene.plugins.fbx.objects;
+
+import com.jme3.texture.Texture;
+import com.jme3.texture.Texture2D;
+import com.jme3.texture.Texture.WrapMode;
+import com.jme3.scene.plugins.fbx.SceneLoader;
+import com.jme3.scene.plugins.fbx.file.FbxElement;
+
+public class FbxTexture extends FbxObject {
+
+ String bindType;
+ String filename;
+
+ public Texture texture;
+
+ public FbxTexture(SceneLoader scene, FbxElement element) {
+ super(scene, element);
+ for(FbxElement e : element.children) {
+ switch(e.id) {
+ case "Type":
+ bindType = (String) e.properties.get(0);
+ break;
+ case "FileName":
+ filename = (String) e.properties.get(0);
+ break;
+ }
+ }
+ texture = new Texture2D();
+ texture.setName(name);
+ texture.setWrap(WrapMode.Repeat); // Default FBX wrapping. TODO: Investigate where this is stored (probably, in material)
+ }
+
+ @Override
+ public void link(FbxObject otherObject) {
+ if(otherObject instanceof FbxImage) {
+ FbxImage img = (FbxImage) otherObject;
+ if(img.image == null)
+ return;
+ texture.setImage(img.image);
+ }
+ }
+}