diff --git a/plugins/3d/oce/loadmodel.cpp b/plugins/3d/oce/loadmodel.cpp
index 875fd37b65..b7622083f9 100644
--- a/plugins/3d/oce/loadmodel.cpp
+++ b/plugins/3d/oce/loadmodel.cpp
@@ -20,6 +20,9 @@
  * or you may search the http://www.gnu.org website for the version 2 license,
  * or you may write to the Free Software Foundation, Inc.,
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
+ *
+ *
+ * Some code lifted from FreeCAD, copyright (c) 2018 Zheng, Lei (realthunder) under GPLv2
  */
 
 #include <iostream>
@@ -82,6 +85,7 @@
 
 // log mask for wxLogTrace
 #define MASK_OCE "PLUGIN_OCE"
+//#define DUMP_LABELS
 
 // precision for mesh creation; 0.07 should be good enough for ECAD viewing
 #define USER_PREC (0.14)
@@ -310,23 +314,6 @@ FormatType fileType( const char* aFileName )
 }
 
 
-static wxString getLabelName( const TDF_Label& label )
-{
-    wxString txt;
-    Handle( TDataStd_Name ) name;
-    if( !label.IsNull() && label.FindAttribute( TDataStd_Name::GetID(), name ) )
-    {
-        TCollection_ExtendedString extstr = name->Get();
-        char*                      str = new char[extstr.LengthOfCString() + 1];
-        extstr.ToUTF8CString( str );
-
-        txt = wxString::FromUTF8( str );
-        delete[] str;
-        txt = txt.Trim();
-    }
-    return txt;
-}
-
 /**
  * Gets the absolute tag string for a given label in the form of ##:##:##:##
  *
@@ -357,6 +344,139 @@ void getTag( const TDF_Label& aLabel, std::string& aTag )
 }
 
 
+#ifdef DUMP_LABELS
+static wxString getLabelName( const TDF_Label& aLabel )
+{
+    wxString txt;
+    Handle( TDataStd_Name ) name;
+    if( !aLabel.IsNull() && aLabel.FindAttribute( TDataStd_Name::GetID(), name ) )
+    {
+        TCollection_ExtendedString extstr = name->Get();
+        char*                      str = new char[extstr.LengthOfCString() + 1];
+        extstr.ToUTF8CString( str );
+
+        txt = wxString::FromUTF8( str );
+        delete[] str;
+        txt = txt.Trim();
+    }
+    return txt;
+}
+
+
+/**
+ * Gets a string for a given TopAbs_ShapeEnum element
+ *
+ * @param aShape enum value to convert
+ */
+std::string getShapeName( TopAbs_ShapeEnum aShape )
+{
+    switch( aShape )
+    {
+    case TopAbs_COMPOUND: return "COMPOUND";
+    case TopAbs_COMPSOLID: return "COMPSOLID";
+    case TopAbs_SOLID: return "SOLID";
+    case TopAbs_SHELL: return "SHELL";
+    case TopAbs_FACE: return "FACE";
+    case TopAbs_WIRE: return "WIRE";
+    case TopAbs_EDGE: return "EDGE";
+    case TopAbs_VERTEX: return "VERTEX";
+    case TopAbs_SHAPE: return "SHAPE";
+    }
+
+    return "UNKNOWN";
+}
+
+static int colorFloatToDecimal( float aVal )
+{
+    return aVal * 255;
+}
+
+
+static inline std::ostream& operator<<( std::ostream& aOStream, const Quantity_ColorRGBA& aColor )
+{
+    Quantity_Color rgb = aColor.GetRGB();
+
+    return aOStream << "rgba(" << colorFloatToDecimal( rgb.Red() ) << ","
+                    << colorFloatToDecimal( rgb.Green() ) << ","
+                    << colorFloatToDecimal( rgb.Blue() ) << ","
+                    << colorFloatToDecimal( aColor.Alpha() )
+                    << ")";
+}
+
+
+/**
+ * Gets a string for a given TopAbs_ShapeEnum element
+ *
+ * @param aLabel Label to convert
+ * @param aShapeTool Handle to shape tool being used
+ * @param aColorTool Handle to color tool being used
+ * @param aPregMsg Any prefixed message to insert (used for indentation in dump)
+ */
+static void printLabel( TDF_Label aLabel, Handle( XCAFDoc_ShapeTool ) aShapeTool,
+                        Handle( XCAFDoc_ColorTool ) aColorTool, const char* aPreMsg = nullptr )
+{
+    if( aLabel.IsNull() )
+        return;
+
+    if( !aPreMsg )
+        aPreMsg = "Label: ";
+
+    TCollection_AsciiString entry;
+    TDF_Tool::Entry( aLabel, entry );
+    std::ostringstream ss;
+    ss << aPreMsg << entry << ", " << getLabelName( aLabel )
+       << ( aShapeTool->IsShape( aLabel ) ? ", shape" : "" )
+       << ( aShapeTool->IsTopLevel( aLabel ) ? ", topLevel" : "" )
+       << ( aShapeTool->IsFree( aLabel ) ? ", free" : "" )
+       << ( aShapeTool->IsAssembly( aLabel ) ? ", assembly" : "" )
+       << ( aShapeTool->IsSimpleShape( aLabel ) ? ", simple" : "" )
+       << ( aShapeTool->IsCompound( aLabel ) ? ", compound" : "" )
+       << ( aShapeTool->IsReference( aLabel ) ? ", reference" : "" )
+       << ( aShapeTool->IsComponent( aLabel ) ? ", component" : "" )
+       << ( aShapeTool->IsSubShape( aLabel ) ? ", subshape" : "" );
+
+    if( aShapeTool->IsSubShape( aLabel ) )
+    {
+        auto shape = aShapeTool->GetShape( aLabel );
+        if( !shape.IsNull() )
+            ss << ", " << getShapeName( shape.ShapeType() );
+    }
+
+    if( aShapeTool->IsShape( aLabel ) )
+    {
+        Quantity_ColorRGBA c;
+        if( aColorTool->GetColor( aLabel, XCAFDoc_ColorGen, c ) )
+            ss << ", gc: " << c;
+        if( aColorTool->GetColor( aLabel, XCAFDoc_ColorSurf, c ) )
+            ss << ", sc: " << c;
+        if( aColorTool->GetColor( aLabel, XCAFDoc_ColorCurv, c ) )
+            ss << ", cc: " << c;
+    }
+
+    wxLogTrace( MASK_OCE, ss.str().c_str() );
+}
+
+
+/**
+ * Dumps a label and the entire tree underneath it
+ *
+ * @param aLabel Label to convert
+ * @param aShapeTool Handle to shape tool being used
+ * @param aColorTool Handle to color tool being used
+ * @param aDepth Indentation level to offset labels (used recursively by dumpLabels)
+ */
+static void dumpLabels( TDF_Label aLabel, Handle( XCAFDoc_ShapeTool ) aShapeTool,
+                        Handle( XCAFDoc_ColorTool ) aColorTool, int aDepth = 0 )
+{
+    std::string indent( aDepth * 2, ' ' );
+    printLabel( aLabel, aShapeTool, aColorTool, indent.c_str() );
+    TDF_ChildIterator it;
+    for( it.Initialize( aLabel ); it.More(); it.Next() )
+        dumpLabels( it.Value(), aShapeTool, aColorTool, aDepth + 1 );
+}
+#endif
+
+
 bool getColor( DATA& data, TDF_Label label, Quantity_Color& color )
 {
     while( true )
@@ -576,6 +696,10 @@ SCENEGRAPH* LoadModel( char const* filename )
     data.m_assy = XCAFDoc_DocumentTool::ShapeTool( data.m_doc->Main() );
     data.m_color = XCAFDoc_DocumentTool::ColorTool( data.m_doc->Main() );
 
+    #ifdef DUMP_LABELS
+    dumpLabels( data.m_doc->Main(), data.m_assy, data.m_color );
+    #endif
+
     // retrieve all free shapes
     TDF_LabelSequence frshapes;
     data.m_assy->GetFreeShapes( frshapes );
@@ -856,10 +980,10 @@ bool processLabel( const TDF_Label& aLabel, DATA& aData, SGNODE* aParent,
 
     if( shapeLabel.HasChild() )
     {
+        wxLogTrace( MASK_OCE, "Label %d has children", labelTag );
         TDF_ChildIterator it;
         for( it.Initialize( shapeLabel ); it.More(); it.Next() )
         {
-            wxLogTrace( MASK_OCE, "Processing label %d child....", labelTag );
             if( processLabel( it.Value(), aData, pptr, aItems ) )
                 ret = true;
         }