LCOV - code coverage report
Current view: top level - l10ntools/source - merge.cxx (source / functions) Hit Total Coverage
Test: commit e02a6cb2c3e2b23b203b422e4e0680877f232636 Lines: 0 215 0.0 %
Date: 2014-04-14 Functions: 0 20 0.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include "sal/config.h"
      21             : 
      22             : #include <algorithm>
      23             : #include <fstream>
      24             : #include <string>
      25             : #include <vector>
      26             : 
      27             : #include "export.hxx"
      28             : #include "po.hxx"
      29             : 
      30             : namespace
      31             : {
      32           0 :     static OString lcl_NormalizeFilename(const OString& rFilename)
      33             :     {
      34             :         return rFilename.copy(
      35             :             std::max(
      36           0 :                 rFilename.lastIndexOf( '\\' ),
      37           0 :                 rFilename.lastIndexOf( '/' ))+1);
      38             :     };
      39             : 
      40           0 :     static bool lcl_ReadPoChecked(
      41             :         PoEntry& o_rPoEntry, PoIfstream& rPoFile,
      42             :         const OString& rFileName)
      43             :     {
      44             :         try
      45             :         {
      46           0 :             rPoFile.readEntry( o_rPoEntry );
      47             :         }
      48           0 :         catch( PoIfstream::Exception& aException )
      49             :         {
      50           0 :             if( aException == PoIfstream::INVALIDENTRY )
      51             :             {
      52             :                 printf(
      53             :                     "Warning : %s contains invalid entry\n",
      54           0 :                     rFileName.getStr() );
      55           0 :                 return false;
      56             :             }
      57           0 :         }
      58           0 :         return true;
      59             :     }
      60             : }
      61             : 
      62             : 
      63             : //  class ResData
      64             : 
      65             : 
      66           0 : ResData::ResData( const OString &rGId )
      67             :     :
      68             :     nIdLevel( ID_LEVEL_NULL ),
      69             :     bChild( false ),
      70             :     bChildWithText( false ),
      71             :     bText( false ),
      72             :     bQuickHelpText( false ),
      73             :     bTitle( false ),
      74             :     sGId( rGId ),
      75           0 :     sTextTyp( "Text" )
      76             : {
      77           0 :     sGId = sGId.replaceAll("\r", OString());
      78           0 : }
      79             : 
      80           0 : ResData::ResData( const OString &rGId, const OString &rFilename)
      81             :     :
      82             :     nIdLevel( ID_LEVEL_NULL ),
      83             :     bChild( false ),
      84             :     bChildWithText( false ),
      85             :     bText( false ),
      86             :     bQuickHelpText( false ),
      87             :     bTitle( false ),
      88             :     sGId( rGId ),
      89             :     sFilename( rFilename ),
      90           0 :     sTextTyp( "Text" )
      91             : {
      92           0 :     sGId = sGId.replaceAll("\r", OString());
      93           0 : }
      94             : 
      95             : 
      96             : // class MergeEntrys
      97             : 
      98             : 
      99           0 : bool MergeEntrys::GetText( OString &rReturn,
     100             :     sal_uInt16 nTyp, const OString &nLangIndex, bool bDel )
     101             : {
     102           0 :     bool bReturn = true;
     103           0 :     switch ( nTyp ) {
     104             :         case STRING_TYP_TEXT :
     105           0 :             rReturn = sText[ nLangIndex ];
     106           0 :             if ( bDel )
     107           0 :                 sText[ nLangIndex ] = "";
     108           0 :             bReturn = bTextFirst[ nLangIndex ];
     109           0 :             bTextFirst[ nLangIndex ] = false;
     110           0 :             break;
     111             :         case STRING_TYP_QUICKHELPTEXT :
     112           0 :             rReturn = sQuickHelpText[ nLangIndex ];
     113           0 :             if ( bDel )
     114           0 :                 sQuickHelpText[ nLangIndex ] = "";
     115           0 :             bReturn = bQuickHelpTextFirst[ nLangIndex ];
     116           0 :             bQuickHelpTextFirst[ nLangIndex ] = false;
     117           0 :             break;
     118             :         case STRING_TYP_TITLE :
     119           0 :             rReturn = sTitle[ nLangIndex ];
     120           0 :             if ( bDel )
     121           0 :                 sTitle[ nLangIndex ] = "";
     122           0 :             bReturn = bTitleFirst[ nLangIndex ];
     123           0 :             bTitleFirst[ nLangIndex ] = false;
     124           0 :             break;
     125             :     }
     126           0 :     return bReturn;
     127             : }
     128             : 
     129             : 
     130           0 : OString MergeEntrys::GetQTZText(const ResData& rResData, const OString& rOrigText)
     131             : {
     132           0 :     const OString sFilename = rResData.sFilename.copy(rResData.sFilename.lastIndexOf('/')+1);
     133             :     const OString sKey =
     134           0 :         PoEntry::genKeyId(sFilename + rResData.sGId + rResData.sId + rResData.sResTyp + rOrigText);
     135           0 :     return sKey + "||" + rOrigText;
     136             : }
     137             : 
     138             : 
     139             : // class MergeDataHashMap
     140             : 
     141             : 
     142           0 : std::pair<MergeDataHashMap::iterator,bool> MergeDataHashMap::insert(const OString& rKey, MergeData* pMergeData)
     143             : {
     144           0 :     std::pair<iterator,bool> aTemp = m_aHashMap.insert(HashMap_t::value_type( rKey, pMergeData ));
     145           0 :     if( m_aHashMap.size() == 1 )
     146             :     {
     147             :         // When first insert, set an iterator to the first element
     148           0 :         aFirstInOrder = aTemp.first;
     149             :     }
     150             :     else
     151             :     {
     152             :         // Define insertion order by setting an iterator to the next element.
     153           0 :         aLastInsertion->second->m_aNextData = aTemp.first;
     154             :     }
     155           0 :     aLastInsertion = aTemp.first;
     156           0 :     return aTemp;
     157             : }
     158             : 
     159           0 : MergeDataHashMap::iterator MergeDataHashMap::find(const OString& rKey)
     160             : {
     161           0 :     iterator aHint = m_aHashMap.end();
     162             : 
     163             :     // Add a hint
     164           0 :     if( bFirstSearch && !m_aHashMap.empty() )
     165             :     {
     166           0 :         aHint = aFirstInOrder;
     167             :     }
     168           0 :     else if( aLastFound == aLastInsertion )
     169             :     {
     170             :         // Next to the last element is the first element
     171           0 :         aHint = aFirstInOrder;
     172             :     }
     173           0 :     else if( aLastFound != m_aHashMap.end() && aLastFound != aLastInsertion )
     174             :     {
     175           0 :         aHint = aLastFound->second->m_aNextData;
     176             :     }
     177             : 
     178             :     // If hint works than no need for search
     179           0 :     if( aHint != m_aHashMap.end() && aHint->first == rKey )
     180             :     {
     181           0 :         aLastFound = aHint;
     182             :     }
     183             :     else
     184             :     {
     185           0 :         aLastFound = m_aHashMap.find(rKey);
     186             :     }
     187             : 
     188           0 :     bFirstSearch = false;
     189           0 :     return aLastFound;
     190             : }
     191             : 
     192             : 
     193             : // class MergeData
     194             : 
     195             : 
     196           0 : MergeData::MergeData(
     197             :     const OString &rTyp, const OString &rGID,
     198             :     const OString &rLID , const OString &rFilename )
     199             :     : sTyp( rTyp ),
     200             :     sGID( rGID ),
     201             :     sLID( rLID ) ,
     202             :     sFilename( rFilename ),
     203           0 :     pMergeEntrys( new MergeEntrys() )
     204             : {
     205           0 : }
     206             : 
     207           0 : MergeData::~MergeData()
     208             : {
     209           0 :     delete pMergeEntrys;
     210           0 : }
     211             : 
     212           0 : MergeEntrys* MergeData::GetMergeEntries()
     213             : {
     214           0 :     return pMergeEntrys;
     215             : }
     216             : 
     217           0 : bool MergeData::operator==( ResData *pData )
     218             : {
     219           0 :     return pData->sId == sLID && pData->sGId == sGID
     220           0 :         && pData->sResTyp.equalsIgnoreAsciiCase(sTyp);
     221             : }
     222             : 
     223             : 
     224             : // class MergeDataFile
     225             : 
     226             : 
     227           0 : MergeDataFile::MergeDataFile(
     228             :     const OString &rFileName, const OString &rFile,
     229           0 :     bool bCaseSensitive, bool bWithQtz )
     230             : {
     231           0 :     OString sEnableReleaseBuild(getenv("ENABLE_RELEASE_BUILD"));
     232             : 
     233           0 :     std::ifstream aInputStream( rFileName.getStr() );
     234           0 :     if ( !aInputStream.is_open() )
     235             :     {
     236           0 :         printf("Warning : Can't open po path container file\n");
     237           0 :         return;
     238             :     }
     239           0 :     std::string sPoFile;
     240           0 :     aInputStream >> sPoFile;
     241           0 :     bool bFirstLang = true;
     242           0 :     while( !aInputStream.eof() )
     243             :     {
     244           0 :         bool bSkipCurrentPOFile = false;
     245           0 :         const OString sFileName( lcl_NormalizeFilename(rFile) );
     246           0 :         const bool bReadAll = sFileName.isEmpty();
     247           0 :         const OString sPoFileName(sPoFile.data(), sPoFile.length());
     248           0 :         PoIfstream aPoInput;
     249           0 :         aPoInput.open( sPoFileName );
     250           0 :         if ( !aPoInput.isOpen() )
     251             :         {
     252           0 :             printf( "Warning : Can't open %s\n", sPoFileName.getStr() );
     253           0 :             return;
     254             :         }
     255             : 
     256           0 :         OString sLang;
     257             :         //Get language id from path
     258             :         {
     259           0 :             const OString sTransSource("translations/source/");
     260             :             const sal_Int32 nStart =
     261           0 :                 sPoFileName.indexOf(sTransSource)+sTransSource.getLength();
     262             :             const sal_Int32 nCount =
     263           0 :                 sPoFileName.indexOf('/',nStart) - nStart;
     264           0 :             sLang = sPoFileName.copy(nStart,nCount);
     265             :         }
     266           0 :         aLanguageSet.insert( sLang );
     267           0 :         PoEntry aNextPo;
     268           0 :         do
     269             :         {
     270           0 :             if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) )
     271             :             {
     272           0 :                 bSkipCurrentPOFile = true;
     273           0 :                 break;
     274             :             }
     275           0 :         } while( !aPoInput.eof() && aNextPo.getSourceFile() != sFileName && !bReadAll );
     276           0 :         while( !aPoInput.eof() && (aNextPo.getSourceFile() == sFileName || bReadAll ) && !bSkipCurrentPOFile )
     277             :         {
     278           0 :             PoEntry aActPo( aNextPo );
     279             : 
     280           0 :             bool bInSameComp = false;
     281           0 :             OString sText;
     282           0 :             OString sQHText;
     283           0 :             OString sTitle;
     284           0 :             OString sExText;
     285           0 :             OString sExQHText;
     286           0 :             OString sExTitle;
     287           0 :             do
     288             :             {
     289           0 :                 if( bInSameComp )
     290           0 :                     aActPo = aNextPo;
     291           0 :                 OString sTemp = aActPo.getMsgStr();
     292           0 :                 if( aActPo.isFuzzy() || sTemp.isEmpty() )
     293           0 :                     sTemp = aActPo.getMsgId();
     294           0 :                 switch( aActPo.getType() )
     295             :                 {
     296             :                     case PoEntry::TTEXT:
     297           0 :                         sText = sTemp;
     298           0 :                         sExText = aActPo.getMsgId();
     299           0 :                         break;
     300             :                     case PoEntry::TQUICKHELPTEXT:
     301           0 :                         sQHText = sTemp;
     302           0 :                         sExQHText = aActPo.getMsgId();
     303           0 :                         break;
     304             :                     case PoEntry::TTITLE:
     305           0 :                         sTitle = sTemp;
     306           0 :                         sExTitle = aActPo.getMsgId();
     307           0 :                         break;
     308             :                 }
     309           0 :                 if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) )
     310             :                 {
     311           0 :                     bSkipCurrentPOFile = true;
     312           0 :                     break;
     313           0 :                 }
     314           0 :             } while( !aPoInput.eof() &&
     315             :                 ( bInSameComp = PoEntry::IsInSameComp(aActPo, aNextPo) ) );
     316             : 
     317             :             InsertEntry(
     318             :                 aActPo.getResourceType(), aActPo.getGroupId(),
     319             :                 aActPo.getLocalId(), sLang, sText,
     320             :                 sQHText, sTitle, aActPo.getSourceFile(),
     321           0 :                 bFirstLang, bCaseSensitive );
     322             : 
     323           0 :             if( bFirstLang && bWithQtz &&
     324           0 :                 !sEnableReleaseBuild.equals("TRUE") )
     325             :             {
     326           0 :                 aLanguageSet.insert("qtz");
     327             :                 InsertEntry(
     328             :                     aActPo.getResourceType(), aActPo.getGroupId(),
     329             :                     aActPo.getLocalId(), "qtz",
     330             :                     sExText, sExQHText,
     331             :                     sExTitle, aActPo.getSourceFile(),
     332           0 :                     false, bCaseSensitive );
     333             :             }
     334           0 :         }
     335           0 :         aPoInput.close();
     336           0 :         aInputStream >> sPoFile;
     337           0 :         bFirstLang = false;
     338           0 :     }
     339           0 :     aInputStream.close();
     340             : }
     341             : 
     342           0 : MergeDataFile::~MergeDataFile()
     343             : {
     344           0 :     for (MergeDataHashMap::iterator aI = aMap.begin(), aEnd = aMap.end(); aI != aEnd; ++aI)
     345           0 :         delete aI->second;
     346           0 : }
     347             : 
     348           0 : std::vector<OString> MergeDataFile::GetLanguages() const
     349             : {
     350           0 :     return std::vector<OString>(aLanguageSet.begin(),aLanguageSet.end());
     351             : }
     352             : 
     353           0 : MergeData *MergeDataFile::GetMergeData( ResData *pResData , bool bCaseSensitive )
     354             : {
     355           0 :     OString sOldG = pResData->sGId;
     356           0 :     OString sOldL = pResData->sId;
     357           0 :     OString sGID = pResData->sGId;
     358           0 :     OString sLID;
     359           0 :     if (sGID.isEmpty())
     360           0 :         sGID = pResData->sId;
     361             :     else
     362           0 :         sLID = pResData->sId;
     363           0 :     pResData->sGId = sGID;
     364           0 :     pResData->sId = sLID;
     365             : 
     366           0 :     OString sKey = CreateKey( pResData->sResTyp , pResData->sGId , pResData->sId , pResData->sFilename , bCaseSensitive );
     367             : 
     368           0 :     MergeDataHashMap::const_iterator mit = aMap.find( sKey );
     369           0 :     if(mit != aMap.end())
     370             :     {
     371           0 :         pResData->sGId = sOldG;
     372           0 :         pResData->sId = sOldL;
     373           0 :         return mit->second;
     374             :     }
     375           0 :     pResData->sGId = sOldG;
     376           0 :     pResData->sId = sOldL;
     377           0 :     return NULL;
     378             : }
     379             : 
     380           0 : MergeEntrys *MergeDataFile::GetMergeEntrys( ResData *pResData )
     381             : {
     382             :     // search for requested MergeEntrys
     383           0 :     MergeData *pData = GetMergeData( pResData );
     384           0 :     if ( pData )
     385           0 :         return pData->GetMergeEntries();
     386           0 :     return NULL;
     387             : }
     388             : 
     389           0 : MergeEntrys *MergeDataFile::GetMergeEntrysCaseSensitive( ResData *pResData )
     390             : {
     391             :     // search for requested MergeEntrys
     392           0 :     MergeData *pData = GetMergeData( pResData , true );
     393           0 :     if ( pData )
     394           0 :         return pData->GetMergeEntries();
     395           0 :     return NULL;
     396             : }
     397             : 
     398           0 : void MergeDataFile::InsertEntry(
     399             :     const OString &rTYP, const OString &rGID,
     400             :     const OString &rLID, const OString &nLANG,
     401             :     const OString &rTEXT, const OString &rQHTEXT,
     402             :     const OString &rTITLE, const OString &rInFilename,
     403             :     bool bFirstLang, bool bCaseSensitive )
     404             : {
     405           0 :     MergeData *pData = 0;
     406             : 
     407             :     // search for MergeData
     408           0 :     OString sKey = CreateKey(rTYP , rGID , rLID , rInFilename , bCaseSensitive);
     409             : 
     410           0 :     if( !bFirstLang )
     411             :     {
     412           0 :         MergeDataHashMap::const_iterator mit = aMap.find( sKey );
     413           0 :         if(mit != aMap.end())
     414           0 :             pData = mit->second;
     415             : 
     416             :     }
     417             : 
     418           0 :     if( !pData )
     419             :     {
     420           0 :         pData = new MergeData( rTYP, rGID, rLID, rInFilename );
     421           0 :         aMap.insert( sKey, pData );
     422             :     }
     423             : 
     424             : 
     425             :     // insert the cur string
     426           0 :     MergeEntrys *pMergeEntrys = pData->GetMergeEntries();
     427           0 :     if( nLANG =="qtz" )
     428             :     {
     429           0 :         const OString sTemp = rInFilename + rGID + rLID + rTYP;
     430             :         pMergeEntrys->InsertEntry(
     431             :             nLANG,
     432           0 :             rTEXT.isEmpty()? rTEXT : PoEntry::genKeyId(sTemp + rTEXT) + "||" + rTEXT,
     433           0 :             rQHTEXT.isEmpty()? rQHTEXT : PoEntry::genKeyId(sTemp + rQHTEXT) + "||" + rQHTEXT,
     434           0 :             rTITLE.isEmpty()? rTITLE : PoEntry::genKeyId(sTemp + rTITLE) + "||" + rTITLE );
     435             :     }
     436             :     else
     437             :     {
     438           0 :         pMergeEntrys->InsertEntry( nLANG , rTEXT, rQHTEXT, rTITLE );
     439           0 :     }
     440           0 : }
     441             : 
     442           0 : OString MergeDataFile::CreateKey(const OString& rTYP, const OString& rGID,
     443             :     const OString& rLID, const OString& rFilename, bool bCaseSensitive)
     444             : {
     445           0 :     static const OString sStroke('-');
     446           0 :     OString sKey( rTYP );
     447           0 :     sKey += sStroke;
     448           0 :     sKey += rGID;
     449           0 :     sKey += sStroke;
     450           0 :     sKey += rLID;
     451           0 :     sKey += sStroke;
     452           0 :     sKey += lcl_NormalizeFilename(rFilename);
     453             :     OSL_TRACE("created key: %s", sKey.getStr());
     454           0 :     if(bCaseSensitive)
     455           0 :         return sKey;         // officecfg case sensitive identifier
     456           0 :     return sKey.toAsciiUpperCase();
     457             : }
     458             : 
     459             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10