فهرست منبع

FBXLoader: add class to dump FBX binary files as text

shadowislord 10 سال پیش
والد
کامیت
d55a859240
1فایلهای تغییر یافته به همراه224 افزوده شده و 0 حذف شده
  1. 224 0
      jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/file/FBXDump.java

+ 224 - 0
jme3-plugins/src/fbx/java/com/jme3/scene/plugins/fbx/file/FBXDump.java

@@ -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 + "}");
+        }
+    }
+}