/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2013 Tuomas Vaherkoski * Copyright (C) 2012 Jean-Pierre Charras, jp.charras@wanadoo.fr * Copyright (C) 2011 Wayne Stambaugh * Copyright (C) 1992-2011 KiCad Developers, see AUTHORS.txt for contributors. * * 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 */ /** * @file vrmlmodelparser.cpp */ #include #include #include #include #include #include "3d_struct.h" #include "modelparsers.h" // separator chars static const char* sep_chars = " \t\n\r"; VRML_MODEL_PARSER::VRML_MODEL_PARSER( S3D_MASTER* aMaster ) : S3D_MODEL_PARSER( aMaster ) {} VRML_MODEL_PARSER::~VRML_MODEL_PARSER() {} void VRML_MODEL_PARSER::Load( const wxString aFilename ) { char line[1024], * text; FILE* file; int LineNum = 0; file = wxFopen( aFilename, wxT( "rt" ) ); if( file == NULL ) { return; } // Switch the locale to standard C (needed to print floating point numbers like 1.3) SetLocaleTo_C_standard(); while( GetLine( file, line, &LineNum, 512 ) ) { text = strtok( line, sep_chars ); if ( text == NULL ) continue; if( stricmp( text, "DEF" ) == 0 || stricmp( text, "Group" ) == 0 ) { while( GetLine( file, line, &LineNum, 512 ) ) { text = strtok( line, sep_chars ); if( text == NULL ) continue; if( *text == '}' ) break; if( stricmp( text, "children" ) == 0 ) { readChildren( file, &LineNum ); } } } } fclose( file ); SetLocaleTo_Default(); // revert to the current locale } int VRML_MODEL_PARSER::readMaterial( FILE* file, int* LineNum ) { char line[512], * text, * command; wxString mat_name; S3D_MATERIAL* material = NULL; command = strtok( NULL, sep_chars ); text = strtok( NULL, sep_chars ); mat_name = FROM_UTF8( text ); if( stricmp( command, "USE" ) == 0 ) { for( material = GetMaster()->m_Materials; material; material = material->Next() ) { if( material->m_Name == mat_name ) { material->SetMaterial(); return 1; } } DBG( printf( "ReadMaterial error: material not found\n" ) ); return 0; } if( stricmp( command, "DEF" ) == 0 || stricmp( command, "Material") == 0) { material = new S3D_MATERIAL( GetMaster(), mat_name ); GetMaster()->Insert( material ); while( GetLine( file, line, LineNum, 512 ) ) { text = strtok( line, sep_chars ); if( text == NULL ) continue; if( text[0] == '}' ) { material->SetMaterial(); return 0; } if( stricmp( text, "diffuseColor" ) == 0 ) { text = strtok( NULL, sep_chars ); material->m_DiffuseColor.x = atof( text ); text = strtok( NULL, sep_chars ); material->m_DiffuseColor.y = atof( text ); text = strtok( NULL, sep_chars ); material->m_DiffuseColor.z = atof( text ); } else if( stricmp( text, "emissiveColor" ) == 0 ) { text = strtok( NULL, sep_chars ); material->m_EmissiveColor.x = atof( text ); text = strtok( NULL, sep_chars ); material->m_EmissiveColor.y = atof( text ); text = strtok( NULL, sep_chars ); material->m_EmissiveColor.z = atof( text ); } else if( strnicmp( text, "specularColor", 13 ) == 0 ) { text = strtok( NULL, sep_chars ); material->m_SpecularColor.x = atof( text ); text = strtok( NULL, sep_chars ); material->m_SpecularColor.y = atof( text ); text = strtok( NULL, sep_chars ); material->m_SpecularColor.z = atof( text ); } else if( strnicmp( text, "ambientIntensity", 16 ) == 0 ) { text = strtok( NULL, sep_chars ); material->m_AmbientIntensity = atof( text ); } else if( strnicmp( text, "transparency", 12 ) == 0 ) { text = strtok( NULL, sep_chars ); material->m_Transparency = atof( text ); } else if( strnicmp( text, "shininess", 9 ) == 0 ) { text = strtok( NULL, sep_chars ); material->m_Shininess = atof( text ); } } } return -1; } int VRML_MODEL_PARSER::readChildren( FILE* file, int* LineNum ) { char line[1024], * text; while( GetLine( file, line, LineNum, 512 ) ) { text = strtok( line, sep_chars ); if( *text == ']' ) return 0; if( *text == ',' ) continue; if( stricmp( text, "Shape" ) == 0 ) { readShape( file, LineNum ); } else { DBG( printf( "ReadChildren error line %d <%s> \n", *LineNum, text ) ); break; } } return 1; } int VRML_MODEL_PARSER::readShape( FILE* file, int* LineNum ) { char line[1024], * text; int err = 1; while( GetLine( file, line, LineNum, 512 ) ) { text = strtok( line, sep_chars ); if( *text == '}' ) { err = 0; break; } if( stricmp( text, "appearance" ) == 0 ) { readAppearance( file, LineNum ); } else if( stricmp( text, "geometry" ) == 0 ) { readGeometry( file, LineNum ); } else { DBG( printf( "ReadShape error line %d <%s> \n", *LineNum, text ) ); break; } } return err; } int VRML_MODEL_PARSER::readAppearance( FILE* file, int* LineNum ) { char line[1024], * text; int err = 1; while( GetLine( file, line, LineNum, 512 ) ) { text = strtok( line, sep_chars ); if( *text == '}' ) { err = 0; break; } if( stricmp( text, "material" ) == 0 ) { readMaterial( file, LineNum ); } else { DBG( printf( "ReadAppearance error line %d <%s> \n", *LineNum, text ) ); break; } } return err; } void VRML_MODEL_PARSER::readCoordsList( FILE* file, char* text_buffer, std::vector< double >& aList, int* LineNum ) { unsigned int ii = 0, jj = 0; char* text; bool HasData = false; bool StartData = false; bool EndNode = false; char string_num[512]; text = text_buffer; while( !EndNode ) { if( *text == 0 ) // Needs data ! { text = text_buffer; GetLine( file, text_buffer, LineNum, 512 ); } while( !EndNode && *text ) { switch( *text ) { case '[': StartData = true; jj = 0; string_num[jj] = 0; break; case '}': EndNode = true; break; case ']': case '\t': case ' ': case ',': jj = 0; if( !StartData || !HasData ) break; aList.push_back( atof( string_num ) ); string_num[jj] = 0; ii++; HasData = false; if( *text == ']' ) { StartData = false; } break; default: if( !StartData ) break; if( jj >= sizeof( string_num ) ) break; string_num[jj] = *text; jj++; string_num[jj] = 0; HasData = true; break; } text++; } } } int VRML_MODEL_PARSER::readGeometry( FILE* file, int* LineNum ) { char line[1024], buffer[1024], * text; int err = 1; std::vector< double > points; std::vector< double > list; double vrmlunits_to_3Dunits = g_Parm_3D_Visu.m_BiuTo3Dunits * UNITS3D_TO_UNITSPCB; while( GetLine( file, line, LineNum, 512 ) ) { strcpy( buffer, line ); text = strtok( buffer, sep_chars ); if( text == NULL ) continue; if( *text == '}' ) { err = 0; break; } if( stricmp( text, "normalPerVertex" ) == 0 ) { text = strtok( NULL, " ,\t\n\r" ); if( text && stricmp( text, "true" ) == 0 ) { } else { } continue; } if( stricmp( text, "colorPerVertex" ) == 0 ) { text = strtok( NULL, " ,\t\n\r" ); if( text && stricmp( text, "true" ) == 0 ) { } else { } continue; } if( stricmp( text, "normal" ) == 0 ) { readCoordsList( file, line, list, LineNum ); list.clear(); continue; } if( stricmp( text, "normalIndex" ) == 0 ) { while( GetLine( file, line, LineNum, 512 ) ) { text = strtok( line, " ,\t\n\r" ); while( text ) { if( *text == ']' ) break; text = strtok( NULL, " ,\t\n\r" ); } if( text && (*text == ']') ) break; } continue; } if( stricmp( text, "color" ) == 0 ) { readCoordsList( file, line, list, LineNum ); list.clear(); continue; } if( stricmp( text, "solid" ) == 0 ) { // ignore solid continue; } if( stricmp( text, "colorIndex" ) == 0 ) { while( GetLine( file, line, LineNum, 512 ) ) { text = strtok( line, " ,\t\n\r" ); while( text ) { if( *text == ']' ) break; text = strtok( NULL, " ,\t\n\r" ); } if( text && (*text == ']') ) break; } continue; } if( stricmp( text, "coord" ) == 0 ) { readCoordsList( file, line, points, LineNum ); } else if( stricmp( text, "coordIndex" ) == 0 ) { if( points.size() < 3 || points.size() % 3 != 0 ) { wxLogError( wxT( "3D geometry read error <%s> at line %d." ), GetChars( FROM_UTF8( text ) ), *LineNum ); err = 1; break; } std::vector< int > coordIndex; std::vector< S3D_VERTEX > vertices; while( GetLine( file, line, LineNum, 512 ) ) { int jj; text = strtok( line, " ,\t\n\r" ); while( text ) { if( *text == ']' ) break; jj = atoi( text ); if( jj < 0 ) { for( jj = 0; jj < (int) coordIndex.size(); jj++ ) { int kk = coordIndex[jj] * 3; if( (kk < 0) || ((kk + 3) > (int)points.size()) ) { wxLogError( wxT( "3D geometry index read error <%s> at line %d." ), GetChars( FROM_UTF8( text ) ), *LineNum ); err = 1; break; } S3D_VERTEX vertex; vertex.x = points[kk]; vertex.y = points[kk + 1]; vertex.z = points[kk + 2]; vertices.push_back( vertex ); } if( GetMaster()->IsOpenGlAllowed() ) { GetMaster()->ObjectCoordsTo3DUnits( vertices ); TransfertToGLlist( vertices, vrmlunits_to_3Dunits ); } vertices.clear(); coordIndex.clear(); } else { coordIndex.push_back( jj ); } text = strtok( NULL, " ,\t\n\r" ); } if( text && (*text == ']') ) break; } } else { wxLogError( wxT( "3D geometry read error <%s> at line %d." ), GetChars( FROM_UTF8( text ) ), *LineNum ); err = 1; break; } } return err; }