/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org 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 Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ #include #include #include "lngmerge.hxx" #include using namespace std; using comphelper::string::getToken; // // class LngParser // LngParser::LngParser(const rtl::OString &rLngFile, sal_Bool bUTF8, sal_Bool bULFFormat) : nError( LNG_OK ) , pLines( NULL ) , sSource( rLngFile ) , bDBIsUTF8( bUTF8 ) , bULF( bULFFormat ) { pLines = new LngLineList(); DirEntry aEntry( String( sSource, RTL_TEXTENCODING_ASCII_US )); if ( aEntry.Exists()) { SvFileStream aStream( String( sSource, RTL_TEXTENCODING_ASCII_US ), STREAM_STD_READ ); if ( aStream.IsOpen()) { rtl::OString sLine; bool bFirstLine = true; while ( !aStream.IsEof()) { aStream.ReadLine( sLine ); if( bFirstLine ) { // Always remove UTF8 BOM from the first line Export::RemoveUTF8ByteOrderMarker( sLine ); bFirstLine = false; } pLines->push_back( new ByteString( sLine ) ); } } else nError = LNG_COULD_NOT_OPEN; } else nError = LNG_FILE_NOTFOUND; } LngParser::~LngParser() { for ( size_t i = 0, n = pLines->size(); i < n; i++ ) delete (*pLines)[ i ]; pLines->clear(); delete pLines; } sal_Bool LngParser::CreateSDF(const rtl::OString &rSDFFile, const rtl::OString &rPrj, const rtl::OString &rRoot) { Export::InitLanguages( false ); aLanguages = Export::GetLanguages(); SvFileStream aSDFStream( String( rSDFFile, RTL_TEXTENCODING_ASCII_US ), STREAM_STD_WRITE | STREAM_TRUNC ); if ( !aSDFStream.IsOpen()) { nError = SDF_COULD_NOT_OPEN; } aSDFStream.SetStreamCharSet( RTL_TEXTENCODING_UTF8 ); nError = SDF_OK; DirEntry aEntry( String( sSource, RTL_TEXTENCODING_ASCII_US )); aEntry.ToAbs(); String sFullEntry = aEntry.GetFull(); aEntry += DirEntry( String( "..", RTL_TEXTENCODING_ASCII_US )); aEntry += DirEntry( rRoot ); rtl::OString sPrjEntry(rtl::OUStringToOString(aEntry.GetFull(), osl_getThreadTextEncoding())); rtl::OString sActFileName(rtl::OUStringToOString( sFullEntry.Copy( sPrjEntry.getLength() + 1 ), osl_getThreadTextEncoding())); sActFileName = sActFileName.replace('/', '\\'); size_t nPos = 0; sal_Bool bStart = true; rtl::OString sGroup, sLine; ByteStringHashMap Text; ByteString sID; while( nPos < pLines->size() ) { sLine = *(*pLines)[ nPos++ ]; while( nPos < pLines->size() && !isNextGroup( sGroup , sLine ) ) { ReadLine( sLine , Text ); sID = sGroup; sLine = *(*pLines)[ nPos++ ]; }; if( bStart ) { bStart = false; sID = sGroup; } else { WriteSDF( aSDFStream , Text , rPrj , rRoot , sActFileName , sID ); } } aSDFStream.Close(); return true; } void LngParser::WriteSDF(SvFileStream &aSDFStream, ByteStringHashMap &rText_inout, const rtl::OString &rPrj, const rtl::OString &rRoot, const rtl::OString &rActFileName, const rtl::OString &rID) { sal_Bool bExport = true; if ( bExport ) { ByteString sTimeStamp( Export::GetTimeStamp()); ByteString sCur; for( unsigned int n = 0; n < aLanguages.size(); n++ ){ sCur = aLanguages[ n ]; ByteString sAct = rText_inout[ sCur ]; if ( !sAct.Len() && sCur.Len() ) sAct = rText_inout[ ByteString("en-US") ]; ByteString sOutput( rPrj ); sOutput += "\t"; if (rRoot.getLength()) sOutput += rActFileName; sOutput += "\t0\t"; sOutput += "LngText\t"; sOutput += rID; sOutput += "\t\t\t\t0\t"; sOutput += sCur; sOutput += "\t"; sOutput += sAct; sOutput += "\t\t\t\t"; sOutput += sTimeStamp; aSDFStream.WriteLine( sOutput ); } } } bool LngParser::isNextGroup(rtl::OString &sGroup_out, rtl::OString &sLine_in) { sLine_in = comphelper::string::stripStart(sLine_in, ' '); sLine_in = comphelper::string::stripEnd(sLine_in, ' '); if ((sLine_in[0] == '[') && (sLine_in[sLine_in.getLength() - 1] == ']')) { sGroup_out = getToken(getToken(sLine_in, 1, '['), 0, ']'); sGroup_out = comphelper::string::stripStart(sGroup_out, ' '); sGroup_out = comphelper::string::stripEnd(sGroup_out, ' '); return true; } return false; } void LngParser::ReadLine(const rtl::OString &rLine_in, ByteStringHashMap &rText_inout) { rtl::OString sLang = getToken(rLine_in, 0, '='); sLang = comphelper::string::stripStart(sLang, ' '); sLang = comphelper::string::stripEnd(sLang, ' '); rtl::OString sText = getToken(getToken(rLine_in, 1, '\"'), 0, '\"'); if (!sLang.isEmpty()) rText_inout[ sLang ] = sText; } sal_Bool LngParser::Merge( const rtl::OString &rSDFFile, const rtl::OString &rDestinationFile) { Export::InitLanguages( true ); SvFileStream aDestination( String( rDestinationFile, RTL_TEXTENCODING_ASCII_US ), STREAM_STD_WRITE | STREAM_TRUNC ); if ( !aDestination.IsOpen()) { nError = LNG_COULD_NOT_OPEN; } nError = LNG_OK; MergeDataFile aMergeDataFile( rSDFFile, sSource, sal_False ); ByteString sTmp( Export::sLanguages ); if( sTmp.ToUpperAscii().Equals("ALL") ) Export::SetLanguages( aMergeDataFile.GetLanguages() ); aLanguages = Export::GetLanguages(); size_t nPos = 0; sal_Bool bGroup = sal_False; rtl::OString sGroup; // seek to next group while ( nPos < pLines->size() && !bGroup ) { rtl::OString sLine( *(*pLines)[ nPos ] ); sLine = comphelper::string::stripStart(sLine, ' '); sLine = comphelper::string::stripEnd(sLine, ' '); if (( sLine[0] == '[' ) && ( sLine[sLine.getLength() - 1] == ']' )) { sGroup = getToken(getToken(sLine, 1, '['), 0, ']'); sGroup = comphelper::string::stripStart(sGroup, ' '); sGroup = comphelper::string::stripEnd(sGroup, ' '); bGroup = sal_True; } nPos ++; } while ( nPos < pLines->size()) { ByteStringHashMap Text; ByteString sID( sGroup ); sal_uLong nLastLangPos = 0; ResData *pResData = new ResData( "", sID , sSource ); pResData->sResTyp = "LngText"; PFormEntrys *pEntrys = aMergeDataFile.GetPFormEntrys( pResData ); // read languages bGroup = sal_False; ByteString sLanguagesDone; while ( nPos < pLines->size() && !bGroup ) { ByteString sLine( *(*pLines)[ nPos ] ); sLine = comphelper::string::stripStart(sLine, ' '); sLine = comphelper::string::stripEnd(sLine, ' '); if (( sLine.GetChar( 0 ) == '[' ) && ( sLine.GetChar( sLine.Len() - 1 ) == ']' )) { sGroup = getToken(getToken(sLine, 1, '['), 0, ']'); sGroup = comphelper::string::stripStart(sGroup, ' '); sGroup = comphelper::string::stripEnd(sGroup, ' '); bGroup = sal_True; nPos ++; sLanguagesDone = ""; } else if ( sLine.GetTokenCount( '=' ) > 1 ) { ByteString sLang = getToken(sLine, 0, '='); sLang = comphelper::string::stripStart(sLang, ' '); sLang = comphelper::string::stripEnd(sLang, ' '); ByteString sSearch( ";" ); sSearch += sLang; sSearch += ";"; if (( sLanguagesDone.Search( sSearch ) != STRING_NOTFOUND )) { LngLineList::iterator it = pLines->begin(); ::std::advance( it, nPos ); pLines->erase( it ); } if( bULF && pEntrys ) { if( sLang.Len() ) { ByteString sNewText; pEntrys->GetText( sNewText, STRING_TYP_TEXT, sLang, sal_True ); if ( sNewText.Len()) { ByteString *pLine = (*pLines)[ nPos ]; ByteString sText1( sLang ); sText1 += " = \""; sText1 += sNewText; sText1 += "\""; *pLine = sText1; Text[ sLang ] = sNewText; } } nLastLangPos = nPos; nPos ++; sLanguagesDone += sSearch; } else { nLastLangPos = nPos; nPos ++; sLanguagesDone += sSearch; } } else nPos++; } ByteString sCur; if ( nLastLangPos ) { for(size_t n = 0; n < aLanguages.size(); ++n) { sCur = aLanguages[ n ]; if( !sCur.EqualsIgnoreCaseAscii("en-US") && Text[sCur].isEmpty() && pEntrys ) { ByteString sNewText; pEntrys->GetText( sNewText, STRING_TYP_TEXT, sCur, sal_True ); if (( sNewText.Len()) && !(( sCur.Equals("x-comment") ) && ( sNewText == "-" ))) { ByteString sLine; sLine += sCur; sLine += " = \""; sLine += sNewText; sLine += "\""; nLastLangPos++; nPos++; if ( nLastLangPos < pLines->size() ) { LngLineList::iterator it = pLines->begin(); ::std::advance( it, nLastLangPos ); pLines->insert( it, new ByteString( sLine ) ); } else { pLines->push_back( new ByteString( sLine ) ); } } } } } delete pResData; } for ( size_t i = 0; i < pLines->size(); i++ ) aDestination.WriteLine( *(*pLines)[ i ] ); aDestination.Close(); return sal_True; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */