/* ************************************************************************** description -------------------- copyright : (C) 2000-2003 by Andreas Zehender email : zehender@kde.org ************************************************************************** ************************************************************************** * * * 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. * * * **************************************************************************/ #include "pmparser.h" #include <tqstring.h> #include <tqbuffer.h> #include <klocale.h> #include "pmpart.h" #include "pmdeclare.h" #include "pmerrorflags.h" #include "pmrecursiveobjectiterator.h" #include "pmdebug.h" unsigned int PMParser::s_maxErrors = 30; unsigned int PMParser::s_maxWarnings = 50; PMParser::PMParser( PMPart* part, TQIODevice* dev ) : m_okDeclares( 101 ) { m_pPart = part; m_pDevice = dev; m_bDeviceCreated = false; init( ); } PMParser::PMParser( PMPart* part, const TQByteArray& array ) : m_okDeclares( 101 ) { m_pPart = part; TQBuffer* buffer = new TQBuffer( array ); buffer->open( IO_ReadOnly ); m_pDevice = TQT_TQIODEVICE(buffer); m_bDeviceCreated = true; init( ); } void PMParser::init( ) { m_okDeclares.setAutoDelete( true ); m_pLocalST.setAutoDelete( true ); m_lineNum = -1; m_pResultList = 0; m_errors = 0; m_warnings = 0; m_bFatalError = false; m_shownMessages = 0; m_messages.clear( ); m_pTopParent = 0; m_renamedObjectSymbols.clear( ); m_okDeclares.clear( ); m_pNextCheckDeclare = 0; } PMParser::~PMParser( ) { if( m_bDeviceCreated ) delete m_pDevice; } void PMParser::printMessage( const PMPMessage messageNum ) { if( !( m_shownMessages & messageNum ) ) { m_shownMessages |= messageNum; switch( messageNum ) { case PMMClockDefault: printWarning( i18n( "Using the default value of 0.0 for clock" ) ); break; case PMMClockDeltaDefault: printWarning( i18n( "Using the default value of 1.0 for clock_delta" ) ); break; case PMMSpecialRawComment: m_messages += PMMessage( i18n( "Note: The full povray syntax is not supported yet. " "If you want to add unsupported povray code to the " "scene, you can put this code between the two " "special comments \"//*PMRawBegin\" and " "\"//*PMRawEnd\"." ) ); break; } } } void PMParser::printMessage( const TQString& type, const TQString& msg ) { if( m_lineNum >= 0 ) m_messages += PMMessage( i18n( "Line %1: " ).arg( m_lineNum ) + type + ": " + msg ); else m_messages += PMMessage( type + ": " + msg ); } void PMParser::printError( const TQString& msg ) { if( m_errors < s_maxErrors ) { printMessage( i18n( "Error" ), msg ); m_errors++; } else if( m_errors == s_maxErrors ) { m_messages += PMMessage( i18n( "Maximum of %1 errors reached." ) .arg( s_maxErrors ) ); m_errors++; } } void PMParser::printWarning( const TQString& msg ) { if( m_warnings < s_maxWarnings ) { printMessage( i18n( "Warning" ), msg ); m_warnings++; } else if( m_warnings == s_maxWarnings ) { m_messages += PMMessage( i18n( "Maximum of %1 warnings reached." ) .arg( s_maxWarnings ) ); m_warnings++; } } void PMParser::printExpected( const char c, const char* sValue ) { printError( i18n( "'%1' expected, found token '%2' instead." ) .arg( c ).arg( sValue ) ); } void PMParser::printExpected( const TQString& str, const char* sValue ) { printError( i18n( "'%1' expected, found token '%2' instead." ) .arg( str ).arg( sValue ) ); } void PMParser::printUnexpected( const TQString& str ) { printError( i18n( "Unexpected token '%1'." ).arg( str ) ); } void PMParser::printInfo( const TQString& msg ) { printMessage( i18n( "Info" ), msg ); } int PMParser::errorFlags( ) const { int result = 0; if( errors( ) ) result |= PMEError; if( warnings( ) ) result |= PMEWarning; if( fatal( ) ) result |= PMEFatal; return result; } void PMParser::parse( PMObjectList* list, PMObject* parent, PMObject* after ) { m_pResultList = list; m_pTopParent = parent; m_pAfter = after; // find first item, that can be a declare and can be used as link // for parsed objects. if( parent ) { if( parent->type( ) == "Scene" ) { if( after ) m_pNextCheckDeclare = after; else m_pNextCheckDeclare = 0; } else { PMObject* obj = parent; bool stop = false; // go to parents, until the parent is the scene // (declares can only be inserted as top level objects) do { if( obj->parent( ) ) { if( obj->parent( )->type( ) == "Scene" ) stop = true; else obj = obj->parent( ); } else { obj = 0; stop = true; } } while( obj && !stop ); // now obj is the top level parent of the object, where parsed objects // will be inserted if( obj ) m_pNextCheckDeclare = obj->prevSibling( ); else m_pNextCheckDeclare = 0; } } topParse( ); TQPtrListIterator<PMSymbol> it( m_renamedObjectSymbols ); for( ; it.current( ); ++it ) it.current( )->setRenamedSymbol( 0 ); m_renamedObjectSymbols.clear( ); m_pLocalST.clear( ); if( ( errors( ) || warnings( ) ) && m_pResultList->isEmpty( ) ) setFatalError( ); } bool PMParser::insertChild( PMObject* child, PMObject* parent ) { bool inserted = false; if( parent ) { if( parent->canInsert( child, parent->lastChild( ) ) ) { parent->appendChild( child ); inserted = true; } else { printError( i18n( "Can't insert %1 into %2." ) .arg( child->description( ) ) .arg( parent->description( ) ) ); } } else { if( m_pTopParent ) { if( m_pTopParent->canInsert( child, m_pAfter, m_pResultList ) ) { m_pResultList->append( child ); inserted = true; } else { printError( i18n( "Can't insert %1 into %2." ) .arg( child->description( ) ) .arg( m_pTopParent->description( ) ) ); } } else { // these lines should not be executed // m_pTopParent may not be null m_pResultList->append( child ); inserted = true; } } if( !inserted ) { // insert error // remove all links PMRecursiveObjectIterator rit( child ); PMDeclare* decl = 0; for( ; rit.current( ); ++rit ) { decl = rit.current( )->linkedObject( ); if( decl ) decl->removeLinkedObject( rit.current( ) ); } } return inserted; } void PMParser::checkID( PMDeclare* decl ) { PMSymbolTable* st = m_pPart->symbolTable( ); PMSymbol* s = m_pLocalST.find( decl->id( ) ); if( !s ) s = st->find( decl->id( ) ); if( s ) { PMSymbol* newSym = st->findNewID( s->id( ) + "_", decl ); s->setRenamedSymbol( newSym ); // Symbol can be inserted multiple times // Faster than searching for s and inserting s // if the list does not contain it. m_renamedObjectSymbols.append( s ); if( m_pTopParent ) m_pLocalST.insert( decl->id( ), newSym ); // paste/drop else st->insert( decl->id( ), newSym ); // load file } else { s = new PMSymbol( decl->id( ), decl ); if( m_pTopParent ) m_pLocalST.insert( decl->id( ), s ); // paste/drop else st->insert( decl->id( ), s ); // load file m_okDeclares.insert( decl->id( ), new bool( true ) ); } } void PMParser::checkID( const TQString& id, const PMValue& v ) { PMSymbolTable* st = m_pPart->symbolTable( ); PMSymbol* s = m_pLocalST.find( id ); if( s ) { PMSymbol* newSym = new PMSymbol( st->findNewID( id + "_" ), v ); s->setRenamedSymbol( newSym ); // Symbol can be inserted multiple times // Faster than searching for s and inserting s // if the list does not contain it. m_renamedObjectSymbols.append( s ); if( m_pTopParent ) m_pLocalST.insert( id, newSym ); // paste/drop // values are never inserted into the parts symbol table // else // st->insert( decl->id( ), newSym ); // load file } else { s = new PMSymbol( id, v ); if( m_pTopParent ) m_pLocalST.insert( id, s ); // paste/drop // values are never inserted into the parts symbol table // else // st->insert( decl->id( ), s ); // load file m_okDeclares.insert( id, new bool( true ) ); } } PMDeclare* PMParser::checkLink( const TQString& id ) { PMSymbolTable* t = m_pPart->symbolTable( ); bool ok = false; // is object declared? PMSymbol* s = m_pLocalST.find( id ); if( !s ) s = t->find( id ); if( !s ) printError( i18n( "Undefined object \"%1\"." ).arg( id ) ); else if( s->type( ) != PMSymbol::Object ) printError( i18n( "Undefined object \"%1\"." ).arg( id ) ); else { // the object is declared // is the id already in m_okDeclares bool* lok = m_okDeclares.find( id ); if( lok ) ok = true; else { // the id is not in m_okDeclares PMObject* obj = s->object( ); while( m_pNextCheckDeclare && !ok ) { if( m_pNextCheckDeclare->isA( "Declare" ) ) { PMDeclare* decl = ( PMDeclare* ) m_pNextCheckDeclare; m_okDeclares.insert( decl->id( ), new bool( true ) ); } if( m_pNextCheckDeclare == obj ) ok = true; m_pNextCheckDeclare = m_pNextCheckDeclare->prevSibling( ); } } if( !ok ) printError( i18n( "Object \"%1\" is undefined at that point." ) .arg( id ) ); } if( ok ) { while( s->renamedSymbol( ) ) s = s->renamedSymbol( ); return s->object( ); } return 0; } PMSymbol* PMParser::getSymbol( const TQString& id ) const { PMSymbol* s = m_pLocalST.find( id ); if( !s ) s = m_pPart->symbolTable( )->find( id ); return s; }