1006 lines
22 KiB
C++
1006 lines
22 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2015-2016 Cirilo Bernardo <cirilo.bernardo@gmail.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, you may find one here:
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
* 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
|
|
*/
|
|
|
|
#include <iostream>
|
|
|
|
#include "vrml2_base.h"
|
|
#include "vrml2_transform.h"
|
|
#include "vrml2_shape.h"
|
|
#include "vrml2_appearance.h"
|
|
#include "vrml2_material.h"
|
|
#include "vrml2_faceset.h"
|
|
#include "vrml2_lineset.h"
|
|
#include "vrml2_pointset.h"
|
|
#include "vrml2_coords.h"
|
|
#include "vrml2_norms.h"
|
|
#include "vrml2_color.h"
|
|
#include "vrml2_box.h"
|
|
#include "vrml2_switch.h"
|
|
#include "plugins/3dapi/ifsg_all.h"
|
|
|
|
|
|
WRL2BASE::WRL2BASE() : WRL2NODE()
|
|
{
|
|
m_useInline = false;
|
|
m_Type = WRL2_BASE;
|
|
return;
|
|
}
|
|
|
|
|
|
WRL2BASE::~WRL2BASE()
|
|
{
|
|
return;
|
|
}
|
|
|
|
|
|
// functions inherited from WRL2NODE
|
|
bool WRL2BASE::SetParent( WRL2NODE* aParent )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] attempting to set parent on WRL2BASE node\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
void WRL2BASE::SetEnableInline( bool enable )
|
|
{
|
|
m_useInline = enable;
|
|
return;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::GetEnableInline( void )
|
|
{
|
|
return m_useInline;
|
|
}
|
|
|
|
|
|
SGNODE* WRL2BASE::AddInlineData( const std::string& aName, WRL2INLINE* aObject )
|
|
{
|
|
std::map< std::string, WRL2INLINE* >::iterator dp = m_inlineModels.find( aName );
|
|
// XXX;
|
|
// qwerty;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
SGNODE* WRL2BASE::GetInlineData( const std::string& aName, WRL2INLINE* aObject )
|
|
{
|
|
// XXX;
|
|
// qwerty;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
void WRL2BASE::DelInlineData( const std::string& aName, WRL2INLINE* aObject )
|
|
{
|
|
// XXX;
|
|
// qwerty;
|
|
return;
|
|
}
|
|
|
|
|
|
std::string WRL2BASE::GetName( void )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] attempting to extract name from virtual base node\n";
|
|
#endif
|
|
|
|
return std::string( "" );
|
|
}
|
|
|
|
|
|
bool WRL2BASE::SetName( const std::string& aName )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] attempting to set name on virtual base node\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::Read( WRLPROC& proc )
|
|
{
|
|
if( proc.GetVRMLType() != VRML_V2 )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] no open file or file is not a VRML2 file\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
WRL2NODE* node = NULL;
|
|
|
|
while( ReadNode( proc, this, &node ) && !proc.eof() );
|
|
|
|
if( proc.eof() )
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::isDangling( void )
|
|
{
|
|
// the base node is never dangling
|
|
return false;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::implementUse( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
if( !aParent )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] invoked with NULL parent\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
std::string glob;
|
|
|
|
if( !proc.ReadName( glob ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
WRL2NODE* ref = aParent->FindNode( glob, NULL );
|
|
|
|
// return 'true' - the file may be defective but it may still be somewhat OK
|
|
if( NULL == ref )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [INFO] node '" << glob << "' not found\n";
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
if( !aParent->AddRefNode( ref ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [INFO] failed to add node '" << glob << "' (";
|
|
std::cerr << ref->GetNodeTypeName( ref->GetNodeType() ) << ") to parent of type ";
|
|
std::cerr << aParent->GetNodeTypeName( aParent->GetNodeType() ) << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = ref;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::implementDef( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
if( NULL == aParent )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] invalid parent pointer (NULL)\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
std::string glob;
|
|
WRL2NODE* lnode = NULL;
|
|
|
|
if( !proc.ReadName( glob ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
size_t line, column;
|
|
proc.GetFilePosData( line, column );
|
|
|
|
if( ReadNode( proc, aParent, &lnode ) )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = lnode;
|
|
|
|
if( lnode && !lnode->SetName( glob ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
size_t line, column;
|
|
proc.GetFilePosData( line, column );
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [INFO] bad formatting (invalid name) at line";
|
|
std::cerr << line << ", column " << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::ReadNode( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
// This function reads a node and stores a pointer to it in aNode.
|
|
// A value 'true' is returned if a node is successfully read or,
|
|
// if the node is not supported, successfully discarded. Callers
|
|
// must always check the value of aNode when the function returns
|
|
// 'true' since it will be NULL if the node type is not supported.
|
|
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
if( NULL == aParent )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] invalid parent pointer (NULL)\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
std::string glob;
|
|
WRL2NODES ntype;
|
|
|
|
if( !proc.ReadName( glob ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
if( !proc.eof() )
|
|
{
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
}
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
// Process node name:
|
|
// the names encountered at this point should be one of the
|
|
// built-in node names or one of:
|
|
// DEF, USE
|
|
// PROTO, EXTERNPROTO
|
|
// ROUTE
|
|
// any PROTO or EXTERNPROTO defined name
|
|
// since we do not support PROTO or EXTERNPROTO, any unmatched names are
|
|
// assumed to be defined via PROTO/EXTERNPROTO and deleted according to
|
|
// a typical pattern.
|
|
|
|
if( !glob.compare( "USE" ) )
|
|
{
|
|
if( !implementUse( proc, aParent, aNode ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if( !glob.compare( "DEF" ) )
|
|
{
|
|
if( !implementDef( proc, aParent, aNode ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if( !glob.compare( "PROTO" ) )
|
|
{
|
|
if( !proc.ReadName( glob ) || !proc.DiscardList() )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if( !glob.compare( "EXTERNPROTO" ) )
|
|
{
|
|
if( !proc.ReadName( glob ) || !proc.ReadName( glob ) || !proc.DiscardList() )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if( !glob.compare( "ROUTE" ) )
|
|
{
|
|
if( !proc.ReadGlob( glob ) || !proc.ReadGlob( glob ) || !proc.ReadGlob( glob ) )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << proc.GetError() << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
ntype = getNodeTypeID( glob );
|
|
size_t line = 0;
|
|
size_t column = 0;
|
|
proc.GetFilePosData( line, column );
|
|
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 2 )
|
|
std::cerr << " * [INFO] Processing node '" << glob << "' ID: " << ntype << "\n";
|
|
#endif
|
|
|
|
switch( ntype )
|
|
{
|
|
//
|
|
// items to be implemented:
|
|
//
|
|
case WRL2_APPEARANCE:
|
|
|
|
if( !readAppearance( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_BOX:
|
|
|
|
if( !readBox( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_COLOR:
|
|
|
|
if( !readColor( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_CONE:
|
|
// XXX - IMPLEMENT
|
|
if( !proc.DiscardNode() )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << " * [INFO] FAIL: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
#ifdef DEBUG_VRML2
|
|
else
|
|
{
|
|
std::cerr << " * [INFO] OK: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
|
|
case WRL2_COORDINATE:
|
|
|
|
if( !readCoords( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_CYLINDER:
|
|
// XXX - IMPLEMENT
|
|
if( !proc.DiscardNode() )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << " * [INFO] FAIL: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
#ifdef DEBUG_VRML2
|
|
else
|
|
{
|
|
std::cerr << " * [INFO] OK: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
|
|
case WRL2_ELEVATIONGRID:
|
|
// XXX - IMPLEMENT
|
|
if( !proc.DiscardNode() )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << " * [INFO] FAIL: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
#ifdef DEBUG_VRML2
|
|
else
|
|
{
|
|
std::cerr << " * [INFO] OK: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
|
|
case WRL2_EXTRUSION:
|
|
// XXX - IMPLEMENT
|
|
if( !proc.DiscardNode() )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << " * [INFO] FAIL: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
#ifdef DEBUG_VRML2
|
|
else
|
|
{
|
|
std::cerr << " * [INFO] OK: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
|
|
case WRL2_INDEXEDFACESET:
|
|
|
|
if( !readFaceSet( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_INDEXEDLINESET:
|
|
|
|
if( !readLineSet( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_POINTSET:
|
|
|
|
if( !readPointSet( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_MATERIAL:
|
|
|
|
if( !readMaterial( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_NORMAL:
|
|
|
|
if( !readNorms( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_SHAPE:
|
|
|
|
if( !readShape( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_SPHERE:
|
|
// XXX - IMPLEMENT
|
|
if( !proc.DiscardNode() )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << " * [INFO] FAIL: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
#ifdef DEBUG_VRML2
|
|
else
|
|
{
|
|
std::cerr << " * [INFO] OK: discard " << glob << " node at l" << line << ", c" << column << "\n";
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
|
|
case WRL2_SWITCH:
|
|
|
|
if( !readSwitch( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
case WRL2_TRANSFORM:
|
|
case WRL2_GROUP:
|
|
|
|
if( !readTransform( proc, aParent, aNode ) )
|
|
return false;
|
|
|
|
break;
|
|
|
|
//
|
|
// items not implemented or for optional future implementation:
|
|
//
|
|
case WRL2_ANCHOR:
|
|
case WRL2_AUDIOCLIP:
|
|
case WRL2_BACKGROUND:
|
|
case WRL2_BILLBOARD:
|
|
case WRL2_COLLISION:
|
|
case WRL2_COLORINTERPOLATOR:
|
|
case WRL2_COORDINATEINTERPOLATOR:
|
|
case WRL2_CYLINDERSENSOR:
|
|
case WRL2_DIRECTIONALLIGHT:
|
|
case WRL2_FOG:
|
|
case WRL2_FONTSTYLE:
|
|
case WRL2_IMAGETEXTURE:
|
|
case WRL2_INLINE:
|
|
case WRL2_LOD:
|
|
case WRL2_MOVIETEXTURE:
|
|
case WRL2_NAVIGATIONINFO:
|
|
case WRL2_NORMALINTERPOLATOR:
|
|
case WRL2_ORIENTATIONINTERPOLATOR:
|
|
case WRL2_PIXELTEXTURE:
|
|
case WRL2_PLANESENSOR:
|
|
case WRL2_POINTLIGHT:
|
|
case WRL2_POSITIONINTERPOLATOR:
|
|
case WRL2_PROXIMITYSENSOR:
|
|
case WRL2_SCALARINTERPOLATOR:
|
|
case WRL2_SCRIPT:
|
|
case WRL2_SOUND:
|
|
case WRL2_SPHERESENSOR:
|
|
case WRL2_SPOTLIGHT:
|
|
case WRL2_TEXT:
|
|
case WRL2_TEXTURECOORDINATE:
|
|
case WRL2_TEXTURETRANSFORM:
|
|
case WRL2_TIMESENSOR:
|
|
case WRL2_TOUCHSENSOR:
|
|
case WRL2_VIEWPOINT:
|
|
case WRL2_VISIBILITYSENSOR:
|
|
case WRL2_WORLDINFO:
|
|
case WRL2_INVALID:
|
|
default:
|
|
|
|
proc.GetFilePosData( line, column );
|
|
|
|
if( !proc.DiscardNode() )
|
|
{
|
|
#if defined( DEBUG_VRML2 ) && ( DEBUG_VRML2 > 1 )
|
|
std::cerr << proc.GetError() << "\n";
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [INFO] could not discard node at line " << line;
|
|
std::cerr << ", column " << column << "\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
#ifdef DEBUG_VRML2
|
|
else
|
|
{
|
|
std::cerr << " * [INFO] OK: discard unsupported " << glob << " node at l";
|
|
std::cerr << line << ", c" << column << "\n";
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::Read( WRLPROC& proc, WRL2BASE* aTopNode )
|
|
{
|
|
// this function makes no sense in the base node
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << proc.GetError() << "\n";
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] this method must never be invoked on a WRL2BASE object\n";
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readTransform( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2TRANSFORM* np = new WRL2TRANSFORM( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readShape( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2SHAPE* np = new WRL2SHAPE( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readAppearance( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2APPEARANCE* np = new WRL2APPEARANCE( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readMaterial( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2MATERIAL* np = new WRL2MATERIAL( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readFaceSet( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2FACESET* np = new WRL2FACESET( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readLineSet( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2LINESET* np = new WRL2LINESET( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readPointSet( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2POINTSET* np = new WRL2POINTSET( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readCoords( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2COORDS* np = new WRL2COORDS( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readNorms( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2NORMS* np = new WRL2NORMS( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readColor( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2COLOR* np = new WRL2COLOR( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readBox( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2BOX* np = new WRL2BOX( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool WRL2BASE::readSwitch( WRLPROC& proc, WRL2NODE* aParent, WRL2NODE** aNode )
|
|
{
|
|
if( NULL != aNode )
|
|
*aNode = NULL;
|
|
|
|
WRL2SWITCH* np = new WRL2SWITCH( aParent );
|
|
|
|
if( !np->Read( proc, this ) )
|
|
{
|
|
delete np;
|
|
return false;
|
|
}
|
|
|
|
if( NULL != aNode )
|
|
*aNode = (WRL2NODE*) np;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
SGNODE* WRL2BASE::TranslateToSG( SGNODE* aParent )
|
|
{
|
|
if( m_Children.empty() )
|
|
return NULL;
|
|
|
|
S3D::SGTYPES ptype = S3D::GetSGNodeType( aParent );
|
|
|
|
if( NULL != aParent && ptype != S3D::SGTYPE_SHAPE )
|
|
{
|
|
#ifdef DEBUG_VRML2
|
|
std::cerr << __FILE__ << ": " << __FUNCTION__ << ": " << __LINE__ << "\n";
|
|
std::cerr << " * [BUG] WRL2BASE does not have a Transform parent (parent ID: ";
|
|
std::cerr << ptype << ")\n";
|
|
#endif
|
|
|
|
return NULL;
|
|
}
|
|
|
|
if( m_sgNode )
|
|
{
|
|
if( NULL != aParent )
|
|
{
|
|
if( NULL == S3D::GetSGNodeParent( m_sgNode )
|
|
&& !S3D::AddSGNodeChild( aParent, m_sgNode ) )
|
|
{
|
|
return NULL;
|
|
}
|
|
else if( aParent != S3D::GetSGNodeParent( m_sgNode )
|
|
&& !S3D::AddSGNodeRef( aParent, m_sgNode ) )
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
return m_sgNode;
|
|
}
|
|
|
|
IFSG_TRANSFORM topNode( aParent );
|
|
|
|
std::list< WRL2NODE* >::iterator sC = m_Children.begin();
|
|
std::list< WRL2NODE* >::iterator eC = m_Children.end();
|
|
WRL2NODES type;
|
|
|
|
// Include only Shape and Transform nodes in the top node
|
|
bool test = false; // set to true if there are any subnodes for display
|
|
|
|
while( sC != eC )
|
|
{
|
|
type = (*sC)->GetNodeType();
|
|
|
|
switch( type )
|
|
{
|
|
case WRL2_SHAPE:
|
|
// wrap the shape in a transform
|
|
do
|
|
{
|
|
IFSG_TRANSFORM wrapper( topNode.GetRawPtr() );
|
|
SGNODE* pshape = (*sC)->TranslateToSG( wrapper.GetRawPtr() );
|
|
|
|
if( NULL != pshape )
|
|
test = true;
|
|
else
|
|
wrapper.Destroy();
|
|
|
|
} while( 0 );
|
|
|
|
break;
|
|
|
|
case WRL2_TRANSFORM:
|
|
case WRL2_SWITCH:
|
|
|
|
if( NULL != (*sC)->TranslateToSG( topNode.GetRawPtr() ) )
|
|
test = true;
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
++ sC;
|
|
}
|
|
|
|
if( false == test )
|
|
{
|
|
topNode.Destroy();
|
|
return NULL;
|
|
}
|
|
|
|
m_sgNode = topNode.GetRawPtr();
|
|
|
|
return m_sgNode;
|
|
}
|