afcfb2fb26
Change-Id: I675bd8a966b2c1654cd6547fc380a51925578835
396 lines
12 KiB
C++
396 lines
12 KiB
C++
/* -*- 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 <algorithm>
|
|
#include <fstream>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "export.hxx"
|
|
#include "po.hxx"
|
|
|
|
namespace
|
|
{
|
|
static ::rtl::OString lcl_NormalizeFilename(const ::rtl::OString& rFilename)
|
|
{
|
|
return rFilename.copy(
|
|
std::max(
|
|
rFilename.lastIndexOf( '\\' ),
|
|
rFilename.lastIndexOf( '/' ))+1);
|
|
};
|
|
|
|
static bool lcl_ReadPoChecked(
|
|
PoEntry& o_rPoEntry, PoIfstream& rPoFile,
|
|
const std::string& rFileName)
|
|
{
|
|
try
|
|
{
|
|
rPoFile.readEntry( o_rPoEntry );
|
|
}
|
|
catch( PoIfstream::Exception& aException )
|
|
{
|
|
if( aException == PoIfstream::INVALIDENTRY )
|
|
{
|
|
printf(
|
|
"Warning : %s contains invalid entry\n",
|
|
rFileName.c_str() );
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
|
|
//
|
|
// class PFormEntrys
|
|
//
|
|
|
|
sal_Bool PFormEntrys::GetTransex3Text( rtl::OString &rReturn,
|
|
sal_uInt16 nTyp, const rtl::OString &nLangIndex, sal_Bool bDel )
|
|
{
|
|
sal_Bool rc = GetText( rReturn , nTyp , nLangIndex , bDel );
|
|
for( sal_Int32 idx = 0; idx < rReturn.getLength(); idx++ )
|
|
{
|
|
if( rReturn[idx] == '\"' && ( idx >= 1 ) && rReturn[idx-1] == '\\' )
|
|
{
|
|
rReturn = rReturn.replaceAt( idx-1, 1, rtl::OString() );
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
/*****************************************************************************/
|
|
sal_Bool PFormEntrys::GetText( rtl::OString &rReturn,
|
|
sal_uInt16 nTyp, const rtl::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_HELPTEXT :
|
|
rReturn = sHelpText;
|
|
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()
|
|
{
|
|
}
|
|
|
|
PFormEntrys* MergeData::GetPFormEntries()
|
|
{
|
|
if( aMap.find( rtl::OString(RTL_CONSTASCII_STRINGPARAM("HACK")) ) != aMap.end() )
|
|
return aMap[rtl::OString(RTL_CONSTASCII_STRINGPARAM("HACK"))];
|
|
return NULL;
|
|
}
|
|
|
|
void MergeData::Insert(PFormEntrys* pfEntrys )
|
|
{
|
|
aMap.insert( PFormEntrysHashMap::value_type( rtl::OString(RTL_CONSTASCII_STRINGPARAM("HACK")) , pfEntrys ) );
|
|
}
|
|
|
|
PFormEntrys* MergeData::GetPFObject( const rtl::OString& rPFO )
|
|
{
|
|
if( aMap.find( rtl::OString(RTL_CONSTASCII_STRINGPARAM("HACK")) ) != aMap.end() )
|
|
return aMap[ rPFO ];
|
|
return NULL;
|
|
}
|
|
|
|
sal_Bool MergeData::operator==( ResData *pData )
|
|
{
|
|
return pData->sId == sLID && pData->sGId == sGID
|
|
&& pData->sResTyp.equalsIgnoreAsciiCase(sTyp);
|
|
}
|
|
|
|
//
|
|
// class MergeDataFile
|
|
//
|
|
|
|
MergeDataFile::MergeDataFile(
|
|
const rtl::OString &rFileName,
|
|
const rtl::OString &rFile,
|
|
bool bCaseSensitive)
|
|
{
|
|
std::ifstream aInputStream( rFileName.getStr() );
|
|
if ( !aInputStream.is_open() )
|
|
{
|
|
printf("Warning : Can't open po path container file");
|
|
return;
|
|
}
|
|
std::string sPoFileName;
|
|
aInputStream >> sPoFileName;
|
|
bool bFirstLang = true;
|
|
while( !aInputStream.eof() )
|
|
{
|
|
const OString sHack("HACK");
|
|
const OString sFileName( lcl_NormalizeFilename(rFile) );
|
|
PoIfstream aPoInput;
|
|
aPoInput.open( OString(sPoFileName.data(), sPoFileName.length()) );
|
|
if ( !aPoInput.isOpen() )
|
|
{
|
|
printf( "Warning : Can't open %s\n", sPoFileName.c_str() );
|
|
return;
|
|
}
|
|
PoHeader aPoHeader;
|
|
try
|
|
{
|
|
aPoInput.readHeader( aPoHeader );
|
|
}
|
|
catch( PoIfstream::Exception& aException )
|
|
{
|
|
if( aException == PoIfstream::INVALIDHEADER )
|
|
{
|
|
printf(
|
|
"Warning : %s has invalid header\n",
|
|
sPoFileName.c_str() );
|
|
return;
|
|
}
|
|
}
|
|
OString sLang;
|
|
try
|
|
{
|
|
sLang = aPoHeader.getLanguage();
|
|
}
|
|
catch( PoHeader::Exception& aException )
|
|
{
|
|
if( aException == PoHeader::NOLANG )
|
|
{
|
|
printf(
|
|
"Warning : %s' header not has language specification\n",
|
|
sPoFileName.c_str() );
|
|
return;
|
|
}
|
|
}
|
|
aLanguageSet.insert( sLang );
|
|
PoEntry aNextPo;
|
|
do
|
|
{
|
|
if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) )
|
|
{
|
|
return;
|
|
}
|
|
} while( !aPoInput.eof() && aNextPo.getSourceFile() != sFileName );
|
|
while( !aPoInput.eof() && aNextPo.getSourceFile() == sFileName )
|
|
{
|
|
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.getTransStr();
|
|
if( aActPo.getFuzzy() || sTemp.isEmpty() )
|
|
sTemp = aActPo.getUnTransStr();
|
|
switch( aActPo.getType() )
|
|
{
|
|
case PoEntry::TTEXT:
|
|
sText = sTemp;
|
|
sExText = aActPo.getUnTransStr();
|
|
sQTZText = aActPo.getKeyId();
|
|
break;
|
|
case PoEntry::TQUICKHELPTEXT:
|
|
sQHText = sTemp;
|
|
sExQHText = aActPo.getUnTransStr();
|
|
sQTZQHText = aActPo.getKeyId();
|
|
break;
|
|
case PoEntry::TTITLE:
|
|
sTitle = sTemp;
|
|
sExTitle = aActPo.getUnTransStr();
|
|
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(), sHack, sLang, sText,
|
|
sQHText, sTitle, sFileName, bCaseSensitive );
|
|
|
|
if( bFirstLang )
|
|
{
|
|
aLanguageSet.insert("qtz");
|
|
InsertEntry(
|
|
aActPo.getResourceType(), aActPo.getGroupId(),
|
|
aActPo.getLocalId(), sHack, "qtz",
|
|
sQTZText + "||" + sExText, sQTZQHText + "||" + sExQHText,
|
|
sQTZTitle + "||" + sExTitle, sFileName, bCaseSensitive );
|
|
}
|
|
}
|
|
aPoInput.close();
|
|
aInputStream >> sPoFileName;
|
|
bFirstLang = false;
|
|
}
|
|
aInputStream.close();
|
|
}
|
|
|
|
MergeDataFile::~MergeDataFile()
|
|
{
|
|
for (MergeDataHashMap::iterator aI = aMap.begin(), aEnd = aMap.end(); aI != aEnd; ++aI)
|
|
delete aI->second;
|
|
}
|
|
|
|
std::vector<rtl::OString> MergeDataFile::GetLanguages() const
|
|
{
|
|
return std::vector<rtl::OString>(aLanguageSet.begin(),aLanguageSet.end());
|
|
}
|
|
|
|
MergeData *MergeDataFile::GetMergeData( ResData *pResData , bool bCaseSensitive )
|
|
{
|
|
rtl::OString sOldG = pResData->sGId;
|
|
rtl::OString sOldL = pResData->sId;
|
|
rtl::OString sGID = pResData->sGId;
|
|
rtl::OString sLID;
|
|
if (sGID.isEmpty())
|
|
sGID = pResData->sId;
|
|
else
|
|
sLID = pResData->sId;
|
|
pResData->sGId = sGID;
|
|
pResData->sId = sLID;
|
|
|
|
rtl::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;
|
|
}
|
|
|
|
|
|
PFormEntrys *MergeDataFile::GetPFormEntrys( ResData *pResData )
|
|
{
|
|
// search for requested PFormEntrys
|
|
MergeData *pData = GetMergeData( pResData );
|
|
if ( pData )
|
|
return pData->GetPFormEntries();
|
|
return NULL;
|
|
}
|
|
|
|
PFormEntrys *MergeDataFile::GetPFormEntrysCaseSensitive( ResData *pResData )
|
|
{
|
|
// search for requested PFormEntrys
|
|
MergeData *pData = GetMergeData( pResData , true );
|
|
if ( pData )
|
|
return pData->GetPFormEntries();
|
|
return NULL;
|
|
}
|
|
|
|
void MergeDataFile::InsertEntry(
|
|
const rtl::OString &rTYP, const rtl::OString &rGID,
|
|
const rtl::OString &rLID, const rtl::OString &rPFO,
|
|
const rtl::OString &nLANG, const rtl::OString &rTEXT,
|
|
const rtl::OString &rQHTEXT, const rtl::OString &rTITLE ,
|
|
const rtl::OString &rInFilename , bool bCaseSensitive
|
|
)
|
|
{
|
|
MergeData *pData;
|
|
|
|
// search for MergeData
|
|
rtl::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 ) );
|
|
}
|
|
|
|
PFormEntrys *pFEntrys = 0;
|
|
|
|
// search for PFormEntrys
|
|
pFEntrys = pData->GetPFObject( rPFO );
|
|
if( !pFEntrys )
|
|
{
|
|
// create new PFormEntrys, cause no one exists with current properties
|
|
pFEntrys = new PFormEntrys( rPFO );
|
|
pData->Insert( pFEntrys );
|
|
}
|
|
|
|
// finaly insert the cur string
|
|
pFEntrys->InsertEntry( nLANG , rTEXT, rQHTEXT, rTITLE );
|
|
}
|
|
|
|
rtl::OString MergeDataFile::CreateKey(const rtl::OString& rTYP, const rtl::OString& rGID,
|
|
const rtl::OString& rLID, const rtl::OString& rFilename, bool bCaseSensitive)
|
|
{
|
|
static const ::rtl::OString sStroke('-');
|
|
::rtl::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: */
|