/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include "sal/config.h" #include #include #include #include #include "export.hxx" #include "po.hxx" namespace { static OString lcl_NormalizeFilename(const OString& rFilename) { return rFilename.copy( std::max( rFilename.lastIndexOf( '\\' ), rFilename.lastIndexOf( '/' ))+1); }; static bool lcl_ReadPoChecked( PoEntry& o_rPoEntry, PoIfstream& rPoFile, const OString& rFileName) { try { rPoFile.readEntry( o_rPoEntry ); } catch( PoIfstream::Exception& aException ) { if( aException == PoIfstream::INVALIDENTRY ) { printf( "Warning : %s contains invalid entry\n", rFileName.getStr() ); return false; } } return true; } } // // class ResData // ResData::ResData( const OString &rGId ) : nIdLevel( ID_LEVEL_NULL ), bChild( sal_False ), bChildWithText( sal_False ), bText( sal_False ), bHelpText( sal_False ), bQuickHelpText( sal_False ), bTitle( sal_False ), bList( sal_False ), sGId( rGId ), nTextRefId( REFID_NONE ), nHelpTextRefId( REFID_NONE ), nQuickHelpTextRefId( REFID_NONE ), nTitleRefId( REFID_NONE ), sTextTyp( "Text" ), pStringList( NULL ), pUIEntries( NULL ), pItemList( NULL ), pFilterList( NULL ), pPairedList( NULL ) { sGId = sGId.replaceAll("\r", OString()); } ResData::ResData( const OString &rGId, const OString &rFilename) : nIdLevel( ID_LEVEL_NULL ), bChild( sal_False ), bChildWithText( sal_False ), bText( sal_False ), bHelpText( sal_False ), bQuickHelpText( sal_False ), bTitle( sal_False ), bList( sal_False ), sGId( rGId ), sFilename( rFilename ), nTextRefId( REFID_NONE ), nHelpTextRefId( REFID_NONE ), nQuickHelpTextRefId( REFID_NONE ), nTitleRefId( REFID_NONE ), sTextTyp( "Text" ), pStringList( NULL ), pUIEntries( NULL ), pItemList( NULL ), pFilterList( NULL ), pPairedList( NULL ) { sGId = sGId.replaceAll("\r", OString()); } ResData::~ResData() { if ( pStringList ) { // delete existing res. of type StringList for ( size_t i = 0; i < pStringList->size(); i++ ) { ExportListEntry* test = (*pStringList)[ i ]; if( test != NULL ) delete test; } delete pStringList; } if ( pFilterList ) { // delete existing res. of type FilterList for ( size_t i = 0; i < pFilterList->size(); i++ ) { ExportListEntry* test = (*pFilterList)[ i ]; delete test; } delete pFilterList; } if ( pItemList ) { // delete existing res. of type ItemList for ( size_t i = 0; i < pItemList->size(); i++ ) { ExportListEntry* test = (*pItemList)[ i ]; delete test; } delete pItemList; } if ( pUIEntries ) { // delete existing res. of type UIEntries for ( size_t i = 0; i < pUIEntries->size(); i++ ) { ExportListEntry* test = (*pUIEntries)[ i ]; delete test; } delete pUIEntries; } } // // class MergeEntrys // sal_Bool MergeEntrys::GetText( OString &rReturn, sal_uInt16 nTyp, const OString &nLangIndex, sal_Bool bDel ) { sal_Bool bReturn=sal_True; switch ( nTyp ) { case STRING_TYP_TEXT : rReturn = sText[ nLangIndex ]; if ( bDel ) sText[ nLangIndex ] = ""; bReturn = bTextFirst[ nLangIndex ]; bTextFirst[ nLangIndex ] = sal_False; break; case STRING_TYP_QUICKHELPTEXT : rReturn = sQuickHelpText[ nLangIndex ]; if ( bDel ) sQuickHelpText[ nLangIndex ] = ""; bReturn = bQuickHelpTextFirst[ nLangIndex ]; bQuickHelpTextFirst[ nLangIndex ] = sal_False; break; case STRING_TYP_TITLE : rReturn = sTitle[ nLangIndex ]; if ( bDel ) sTitle[ nLangIndex ] = ""; bReturn = bTitleFirst[ nLangIndex ]; bTitleFirst[ nLangIndex ] = sal_False; break; } return bReturn; } // // class MergeData // MergeData::MergeData( const OString &rTyp, const OString &rGID, const OString &rLID , const OString &rFilename ) : sTyp( rTyp ), sGID( rGID ), sLID( rLID ) , sFilename( rFilename ), pMergeEntrys( new MergeEntrys() ) { } MergeData::~MergeData() { delete pMergeEntrys; } MergeEntrys* MergeData::GetMergeEntries() { return pMergeEntrys; } sal_Bool MergeData::operator==( ResData *pData ) { return pData->sId == sLID && pData->sGId == sGID && pData->sResTyp.equalsIgnoreAsciiCase(sTyp); } // // class MergeDataFile // MergeDataFile::MergeDataFile( const OString &rFileName, const OString &rFile, bool bCaseSensitive, bool bWithQtz ) { std::ifstream aInputStream( rFileName.getStr() ); if ( !aInputStream.is_open() ) { printf("Warning : Can't open po path container file\n"); return; } std::string sPoFile; aInputStream >> sPoFile; bool bFirstLang = true; while( !aInputStream.eof() ) { const OString sFileName( lcl_NormalizeFilename(rFile) ); const bool bReadAll = sFileName.isEmpty(); const OString sPoFileName(sPoFile.data(), sPoFile.length()); PoIfstream aPoInput; aPoInput.open( sPoFileName ); if ( !aPoInput.isOpen() ) { printf( "Warning : Can't open %s\n", sPoFileName.getStr() ); return; } OString sLang; //Get language id from path { const OString sTransSource("translations/source/"); const sal_Int32 nStart = sPoFileName.indexOf(sTransSource)+sTransSource.getLength(); const sal_Int32 nCount = sPoFileName.indexOf("/",nStart) - nStart; sLang = sPoFileName.copy(nStart,nCount); } aLanguageSet.insert( sLang ); PoEntry aNextPo; do { if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) ) { return; } } while( !aPoInput.eof() && aNextPo.getSourceFile() != sFileName && !bReadAll ); while( !aPoInput.eof() && (aNextPo.getSourceFile() == sFileName || bReadAll )) { PoEntry aActPo( aNextPo ); bool bInSameComp = false; OString sText; OString sQHText; OString sTitle; OString sExText; OString sExQHText; OString sExTitle; OString sQTZText; OString sQTZQHText; OString sQTZTitle; do { if( bInSameComp ) aActPo = aNextPo; OString sTemp = aActPo.getMsgStr(); if( aActPo.isFuzzy() || sTemp.isEmpty() ) sTemp = aActPo.getMsgId(); switch( aActPo.getType() ) { case PoEntry::TTEXT: sText = sTemp; sExText = aActPo.getMsgId(); sQTZText = aActPo.getKeyId(); break; case PoEntry::TQUICKHELPTEXT: sQHText = sTemp; sExQHText = aActPo.getMsgId(); sQTZQHText = aActPo.getKeyId(); break; case PoEntry::TTITLE: sTitle = sTemp; sExTitle = aActPo.getMsgId(); sQTZTitle = aActPo.getKeyId(); break; } if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) ) { return; } } while( !aPoInput.eof() && ( bInSameComp = PoEntry::IsInSameComp(aActPo, aNextPo) ) ); InsertEntry( aActPo.getResourceType(), aActPo.getGroupId(), aActPo.getLocalId(), sLang, sText, sQHText, sTitle, aActPo.getSourceFile(), bCaseSensitive ); if( bFirstLang && bWithQtz && ( strcmp(getenv("ENABLE_RELEASE_BUILD"),"TRUE") ) ) { aLanguageSet.insert("qtz"); InsertEntry( aActPo.getResourceType(), aActPo.getGroupId(), aActPo.getLocalId(), "qtz", sQTZText + "||" + sExText, sQTZQHText + "||" + sExQHText, sQTZTitle + "||" + sExTitle, aActPo.getSourceFile(), bCaseSensitive ); } } aPoInput.close(); aInputStream >> sPoFile; bFirstLang = false; } aInputStream.close(); } MergeDataFile::~MergeDataFile() { for (MergeDataHashMap::iterator aI = aMap.begin(), aEnd = aMap.end(); aI != aEnd; ++aI) delete aI->second; } std::vector MergeDataFile::GetLanguages() const { return std::vector(aLanguageSet.begin(),aLanguageSet.end()); } MergeData *MergeDataFile::GetMergeData( ResData *pResData , bool bCaseSensitive ) { OString sOldG = pResData->sGId; OString sOldL = pResData->sId; OString sGID = pResData->sGId; OString sLID; if (sGID.isEmpty()) sGID = pResData->sId; else sLID = pResData->sId; pResData->sGId = sGID; pResData->sId = sLID; OString sKey = CreateKey( pResData->sResTyp , pResData->sGId , pResData->sId , pResData->sFilename , bCaseSensitive ); if(aMap.find( sKey ) != aMap.end()) { pResData->sGId = sOldG; pResData->sId = sOldL; return aMap[ sKey ]; } pResData->sGId = sOldG; pResData->sId = sOldL; return NULL; } MergeEntrys *MergeDataFile::GetMergeEntrys( ResData *pResData ) { // search for requested MergeEntrys MergeData *pData = GetMergeData( pResData ); if ( pData ) return pData->GetMergeEntries(); return NULL; } MergeEntrys *MergeDataFile::GetMergeEntrysCaseSensitive( ResData *pResData ) { // search for requested MergeEntrys MergeData *pData = GetMergeData( pResData , true ); if ( pData ) return pData->GetMergeEntries(); return NULL; } void MergeDataFile::InsertEntry( const OString &rTYP, const OString &rGID, const OString &rLID, const OString &nLANG, const OString &rTEXT, const OString &rQHTEXT, const OString &rTITLE, const OString &rInFilename, bool bCaseSensitive ) { MergeData *pData; // search for MergeData OString sKey = CreateKey(rTYP , rGID , rLID , rInFilename , bCaseSensitive); MergeDataHashMap::const_iterator mit; mit = aMap.find( sKey ); if( mit != aMap.end() ) { pData = mit->second; } else { pData = new MergeData( rTYP, rGID, rLID, rInFilename ); aMap.insert( MergeDataHashMap::value_type( sKey, pData ) ); } // insert the cur string MergeEntrys *pMergeEntrys = pData->GetMergeEntries(); pMergeEntrys->InsertEntry( nLANG , rTEXT, rQHTEXT, rTITLE ); } OString MergeDataFile::CreateKey(const OString& rTYP, const OString& rGID, const OString& rLID, const OString& rFilename, bool bCaseSensitive) { static const OString sStroke('-'); OString sKey( rTYP ); sKey += sStroke; sKey += rGID; sKey += sStroke; sKey += rLID; sKey += sStroke; sKey += lcl_NormalizeFilename(rFilename); OSL_TRACE("created key: %s", sKey.getStr()); if(bCaseSensitive) return sKey; // officecfg case sensitive identifier return sKey.toAsciiUpperCase(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */