FBXLoader: add class to dump FBX binary files as text
This commit is contained in:
parent
92e08b1f2b
commit
d55a859240
@ -0,0 +1,224 @@
|
||||
/*
|
||||
* Copyright (c) 2009-2015 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.file;
|
||||
|
||||
import java.io.OutputStream;
|
||||
import java.io.PrintStream;
|
||||
import java.lang.reflect.Array;
|
||||
import java.text.DecimalFormat;
|
||||
import java.util.HashMap;
|
||||
import java.util.Map;
|
||||
import static org.omg.IOP.IORHelper.id;
|
||||
|
||||
/**
|
||||
* Quick n' dirty dumper of FBX binary files.
|
||||
*
|
||||
* Outputs a format similar to an ASCII FBX file.
|
||||
*
|
||||
* @author Kirill Vainer
|
||||
*/
|
||||
public final class FBXDump {
|
||||
|
||||
private static final DecimalFormat DECIMAL_FORMAT
|
||||
= new DecimalFormat("0.0000000000");
|
||||
|
||||
private FBXDump() { }
|
||||
|
||||
/**
|
||||
* Creates a map between object UIDs and the objects themselves.
|
||||
*
|
||||
* @param file The file to create the mappings for.
|
||||
* @return The UID to object map.
|
||||
*/
|
||||
private static Map<Long, FBXElement> createUidToObjectMap(FBXFile file) {
|
||||
Map<Long, FBXElement> uidToObjectMap = new HashMap<Long, FBXElement>();
|
||||
for (FBXElement rootElement : file.rootElements) {
|
||||
if (rootElement.id.equals("Objects")) {
|
||||
for (FBXElement fbxObj : rootElement.children) {
|
||||
if (fbxObj.propertiesTypes[0] != 'L') {
|
||||
continue; // error
|
||||
}
|
||||
Long uid = (Long) fbxObj.properties.get(0);
|
||||
uidToObjectMap.put(uid, fbxObj);
|
||||
}
|
||||
}
|
||||
}
|
||||
return uidToObjectMap;
|
||||
}
|
||||
|
||||
/**
|
||||
* Dump FBX to standard output.
|
||||
*
|
||||
* @param file the file to dump.
|
||||
*/
|
||||
public static void dumpFBX(FBXFile file) {
|
||||
dumpFBX(file, System.out);
|
||||
}
|
||||
|
||||
/**
|
||||
* Dump FBX to the given output stream.
|
||||
*
|
||||
* @param file the file to dump.
|
||||
* @param out the output stream where to output.
|
||||
*/
|
||||
public static void dumpFBX(FBXFile file, OutputStream out) {
|
||||
Map<Long, FBXElement> uidToObjectMap = createUidToObjectMap(file);
|
||||
PrintStream ps = new PrintStream(out);
|
||||
for (FBXElement rootElement : file.rootElements) {
|
||||
dumpFBXElement(rootElement, ps, 0, uidToObjectMap);
|
||||
}
|
||||
}
|
||||
|
||||
private static String indent(int amount) {
|
||||
return " ".substring(0, amount);
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert FBX string - this replaces all instances of
|
||||
* <code>\x00\x01</code> to "::".
|
||||
*
|
||||
* @param string The string to convert
|
||||
* @return
|
||||
*/
|
||||
private static String convertFBXString(String string) {
|
||||
return string.replaceAll("\u0000\u0001", "::");
|
||||
}
|
||||
|
||||
protected static void dumpFBXProperty(String id, char propertyType,
|
||||
Object property, PrintStream ps,
|
||||
Map<Long, FBXElement> uidToObjectMap) {
|
||||
switch (propertyType) {
|
||||
case 'S':
|
||||
// String
|
||||
String str = (String) property;
|
||||
ps.print("\"" + convertFBXString(str) + "\"");
|
||||
break;
|
||||
case 'R':
|
||||
// RAW data.
|
||||
byte[] bytes = (byte[]) property;
|
||||
ps.print("[");
|
||||
for (int j = 0; j < bytes.length; j++) {
|
||||
ps.print(String.format("%02X", bytes[j] & 0xff));
|
||||
if (j != bytes.length - 1) {
|
||||
ps.print(" ");
|
||||
}
|
||||
}
|
||||
ps.print("]");
|
||||
break;
|
||||
case 'D':
|
||||
case 'F':
|
||||
// Double, Float.
|
||||
if (property instanceof Double) {
|
||||
ps.print(DECIMAL_FORMAT.format((Double)property));
|
||||
} else if (property instanceof Float) {
|
||||
ps.print(DECIMAL_FORMAT.format((Float)property));
|
||||
} else {
|
||||
ps.print(property);
|
||||
}
|
||||
break;
|
||||
case 'I':
|
||||
case 'Y':
|
||||
// Integer, Signed Short.
|
||||
ps.print(property);
|
||||
break;
|
||||
case 'C':
|
||||
// Boolean
|
||||
ps.print((Boolean)property ? "1" : "0");
|
||||
break;
|
||||
case 'L':
|
||||
// Long
|
||||
// If this is a connection, decode UID into object name.
|
||||
if (id.equals("C")) {
|
||||
Long uid = (Long) property;
|
||||
FBXElement element = uidToObjectMap.get(uid);
|
||||
if (element != null) {
|
||||
String name = (String) element.properties.get(1);
|
||||
ps.print("\"" + convertFBXString(name) + "\"");
|
||||
} else {
|
||||
ps.print(property);
|
||||
}
|
||||
} else {
|
||||
ps.print(property);
|
||||
}
|
||||
break;
|
||||
case 'd':
|
||||
case 'i':
|
||||
case 'l':
|
||||
case 'f':
|
||||
// Arrays of things..
|
||||
int length = Array.getLength(property);
|
||||
for (int j = 0; j < length; j++) {
|
||||
Object arrayEntry = Array.get(property, j);
|
||||
dumpFBXProperty(id, Character.toUpperCase(propertyType), arrayEntry, ps, uidToObjectMap);
|
||||
if (j != length - 1) {
|
||||
ps.print(",");
|
||||
}
|
||||
}
|
||||
break;
|
||||
default:
|
||||
throw new UnsupportedOperationException("" + propertyType);
|
||||
}
|
||||
}
|
||||
|
||||
protected static void dumpFBXElement(FBXElement el, PrintStream ps,
|
||||
int indent, Map<Long, FBXElement> uidToObjectMap) {
|
||||
// 4 spaces per tab should be OK.
|
||||
String indentStr = indent(indent * 4);
|
||||
String textId = el.id;
|
||||
|
||||
// Properties are called 'P' and connections are called 'C'.
|
||||
if (el.id.equals("P")) {
|
||||
textId = "Property";
|
||||
} else if (el.id.equals("C")) {
|
||||
textId = "Connect";
|
||||
}
|
||||
|
||||
ps.print(indentStr + textId + ": ");
|
||||
for (int i = 0; i < el.properties.size(); i++) {
|
||||
Object property = el.properties.get(i);
|
||||
char propertyType = el.propertiesTypes[i];
|
||||
dumpFBXProperty(el.id, propertyType, property, ps, uidToObjectMap);
|
||||
if (i != el.properties.size() - 1) {
|
||||
ps.print(", ");
|
||||
}
|
||||
}
|
||||
if (el.children.isEmpty()) {
|
||||
ps.println();
|
||||
} else {
|
||||
ps.println(" {");
|
||||
for (FBXElement childElement : el.children) {
|
||||
dumpFBXElement(childElement, ps, indent + 1, uidToObjectMap);
|
||||
}
|
||||
ps.println(indentStr + "}");
|
||||
}
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user