diff options
author | Caolán McNamara <caolanm@redhat.com> | 2017-06-11 20:56:30 +0100 |
---|---|---|
committer | Caolán McNamara <caolanm@redhat.com> | 2017-07-21 08:20:50 +0100 |
commit | 00657aef09d854c74fb426a935a3e8b1fc390bb0 (patch) | |
tree | fd1a9bb264fe15dcc129498e62060ecd256b1ee7 /tools | |
parent | fa987cbb813cfd729fe490f2f1258b7c8d7fb174 (diff) |
migrate to boost::gettext
* all .ui files go from <interface> to <interface domain="MODULE"> e.g. vcl
* all .src files go away and the english source strings folded into the .hrc as NC_("context", "source string")
* ResMgr is dropped in favour of std::locale imbued by boost::locale::generator pointed at matching
MODULE .mo files
* UIConfig translations are folded into the module .mo, so e.g. UIConfig_cui
goes from l10n target to normal one, so the res/lang.zips of UI files go away
* translation via Translation::get(hrc-define-key, imbued-std::locale)
* python can now be translated with its inbuilt gettext support (we keep the name strings.hrc there
to keep finding the .hrc file uniform) so magic numbers can go away there
* java and starbasic components can be translated via the pre-existing css.resource.StringResourceWithLocation
mechanism
* en-US res files go away, their strings are now the .hrc keys in the source code
* remaining .res files are replaced by .mo files
* in .res/.ui-lang-zip files, the old scheme missing translations of strings
results in inserting the english original so something can be found, now the
standard fallback of using the english original from the source key is used, so
partial translations shrink dramatically in size
* extract .hrc strings with hrcex which backs onto
xgettext -C --add-comments --keyword=NC_:1c,2 --from-code=UTF-8 --no-wrap
* extract .ui strings with uiex which backs onto
xgettext --add-comments --no-wrap
* qtz for gettext translations is generated at runtime as ascii-ified crc32 of
content + "|" + msgid
* [API CHANGE] remove deprecated binary .res resouce loader related uno apis
com::sun::star::resource::OfficeResourceLoader
com::sun::star::resource::XResourceBundleLoader
com::sun::star::resource::XResourceBundle
when translating strings via uno apis
com.sun.star.resource.StringResourceWithLocation
can continue to be used
Change-Id: Ia2594a2672b7301d9c3421fdf31b6cfe7f3f8d0a
Diffstat (limited to 'tools')
-rw-r--r-- | tools/Library_tl.mk | 3 | ||||
-rw-r--r-- | tools/inc/pch/precompiled_tl.hxx | 1 | ||||
-rw-r--r-- | tools/source/rc/rc.cxx | 67 | ||||
-rw-r--r-- | tools/source/rc/resary.cxx | 107 | ||||
-rw-r--r-- | tools/source/rc/resmgr.cxx | 1340 |
5 files changed, 62 insertions, 1456 deletions
diff --git a/tools/Library_tl.mk b/tools/Library_tl.mk index 86c24fdbb984..f24d4a5851a4 100644 --- a/tools/Library_tl.mk +++ b/tools/Library_tl.mk @@ -69,8 +69,6 @@ $(eval $(call gb_Library_add_exception_objects,tl,\ tools/source/memtools/unqidx \ tools/source/misc/cpuid \ tools/source/misc/extendapplicationenvironment \ - tools/source/rc/rc \ - tools/source/rc/resary \ tools/source/rc/resmgr \ tools/source/ref/globname \ tools/source/ref/pstm \ @@ -96,6 +94,7 @@ $(eval $(call gb_Library_add_generated_exception_objects,tl,\ $(eval $(call gb_Library_use_externals,tl,\ boost_headers \ + boost_locale \ zlib \ )) diff --git a/tools/inc/pch/precompiled_tl.hxx b/tools/inc/pch/precompiled_tl.hxx index c906f41f3e52..c986b4c5b79f 100644 --- a/tools/inc/pch/precompiled_tl.hxx +++ b/tools/inc/pch/precompiled_tl.hxx @@ -75,7 +75,6 @@ #include <o3tl/cow_wrapper.hxx> #include <tools/debug.hxx> #include <tools/gen.hxx> -#include <tools/resid.hxx> #include <tools/solar.h> #include <tools/stream.hxx> #include <tools/toolsdllapi.h> diff --git a/tools/source/rc/rc.cxx b/tools/source/rc/rc.cxx deleted file mode 100644 index 2b972a8145f9..000000000000 --- a/tools/source/rc/rc.cxx +++ /dev/null @@ -1,67 +0,0 @@ -/* -*- 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 <string.h> -#include <rtl/ustring.hxx> -#include <tools/date.hxx> -#include <tools/time.hxx> -#include <tools/rcid.h> -#include <tools/resmgr.hxx> - -OUString ResId::toString() const -{ - SetRT( RSC_STRING ); - - ResMgr* pResMgr = GetResMgr(); - - if ( !pResMgr || !pResMgr->GetResource( *this ) ) - { - OUString sRet; - -#if OSL_DEBUG_LEVEL > 0 - sRet = "<resource id " - + OUString::number(static_cast<sal_Int32>(GetId())) - + " not found>"; -#endif - - if( pResMgr ) - pResMgr->PopContext(); - - return sRet; - } - - // String loading - RSHEADER_TYPE * pResHdr = static_cast<RSHEADER_TYPE*>(pResMgr->GetClass()); - - sal_Int32 nStringLen = rtl_str_getLength( reinterpret_cast<char*>(pResHdr+1) ); - OUString sRet(reinterpret_cast<char*>(pResHdr+1), nStringLen, RTL_TEXTENCODING_UTF8); - - sal_uInt32 nSize = sizeof( RSHEADER_TYPE ) - + sal::static_int_cast< sal_uInt32 >(nStringLen) + 1; - nSize += nSize % 2; - pResMgr->Increment( nSize ); - - ResHookProc pImplResHookProc = ResMgr::GetReadStringHook(); - if ( pImplResHookProc ) - sRet = pImplResHookProc(sRet); - return sRet; -} - - -/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ diff --git a/tools/source/rc/resary.cxx b/tools/source/rc/resary.cxx deleted file mode 100644 index 078b5a63b8f7..000000000000 --- a/tools/source/rc/resary.cxx +++ /dev/null @@ -1,107 +0,0 @@ -/* -*- 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 <tools/resary.hxx> -#include <tools/resmgr.hxx> -#include <tools/rcid.h> - -#include <vector> - -namespace { - -struct ImplResStringItem -{ - OUString m_aStr; - sal_IntPtr m_nValue; - - ImplResStringItem( const OUString& rStr, long nValue = 0 ) : - m_aStr(rStr), - m_nValue(nValue) - { } -}; - -} - -struct ResStringArray::Impl -{ - std::vector<ImplResStringItem> m_aStrings; -}; - -ResStringArray::ResStringArray( const ResId& rResId ) : - mpImpl(new Impl) -{ - rResId.SetRT( RSC_STRINGARRAY ); - ResMgr* pMgr = rResId.GetResMgr(); - if( pMgr && pMgr->GetResource( rResId ) ) - { - pMgr->GetClass(); - pMgr->Increment( sizeof( RSHEADER_TYPE ) ); - const sal_uInt32 nItems = pMgr->ReadLong(); - if ( nItems ) - { - mpImpl->m_aStrings.reserve( nItems ); - for ( sal_uInt32 i = 0; i < nItems; i++ ) - { - // load string - mpImpl->m_aStrings.push_back(ImplResStringItem(pMgr->ReadString())); - - // load value - mpImpl->m_aStrings[i].m_nValue = pMgr->ReadLong(); - } - } - } -} - -ResStringArray::~ResStringArray() -{ -} - -OUString ResStringArray::GetString( sal_uInt32 nIndex ) const -{ - return (nIndex < mpImpl->m_aStrings.size()) ? mpImpl->m_aStrings[nIndex].m_aStr : OUString(); -} - -sal_IntPtr ResStringArray::GetValue( sal_uInt32 nIndex ) const -{ - return (nIndex < mpImpl->m_aStrings.size()) ? mpImpl->m_aStrings[nIndex].m_nValue : -1; -} - -sal_uInt32 ResStringArray::Count() const -{ - return sal_uInt32(mpImpl->m_aStrings.size()); -} - -sal_uInt32 ResStringArray::FindIndex( sal_IntPtr nValue ) const -{ - const sal_uInt32 nItems = mpImpl->m_aStrings.size(); - for ( sal_uInt32 i = 0; i < nItems; i++ ) - { - if (mpImpl->m_aStrings[i].m_nValue == nValue) - return i; - } - return RESARRAY_INDEX_NOTFOUND; -} - -sal_uInt32 ResStringArray::AddItem( const OUString& rString, sal_IntPtr nValue ) -{ - mpImpl->m_aStrings.push_back(ImplResStringItem(rString, nValue)); - return mpImpl->m_aStrings.size(); -} - -/* vim:set shiftwidth=4 softtabstop=4 expandtab: */ diff --git a/tools/source/rc/resmgr.cxx b/tools/source/rc/resmgr.cxx index 4876794bd6e4..293a78d24ab9 100644 --- a/tools/source/rc/resmgr.cxx +++ b/tools/source/rc/resmgr.cxx @@ -30,13 +30,13 @@ #include <tools/debug.hxx> #include <tools/stream.hxx> #include <tools/resmgr.hxx> -#include <tools/rcid.h> #include <osl/endian.h> #include <osl/process.h> #include <osl/thread.h> #include <osl/file.hxx> #include <osl/mutex.hxx> #include <osl/signal.h> +#include <rtl/crc.h> #include <rtl/ustrbuf.hxx> #include <rtl/strbuf.hxx> #include <sal/log.hxx> @@ -44,7 +44,9 @@ #include <rtl/bootstrap.hxx> #include <i18nlangtag/languagetag.hxx> #include <i18nlangtag/mslangid.hxx> -#include <tools/simplerm.hxx> + +#include <boost/locale.hpp> +#include <boost/locale/gnu_gettext.hpp> #include <algorithm> #include <list> @@ -53,1317 +55,97 @@ #include <unordered_map> #include <memory> -using namespace osl; - -// for thread safety -static osl::Mutex* pResMgrMutex = nullptr; - -static osl::Mutex& getResMgrMutex() -{ - if( !pResMgrMutex ) - { - osl::Guard<osl::Mutex> aGuard( *osl::Mutex::getGlobalMutex() ); - if( ! pResMgrMutex ) - pResMgrMutex = new osl::Mutex(); - } - return *pResMgrMutex; -} - -struct ImpContent; - -// seem to need this to put a std::pair<RESOURCE_TYPE,sal_uInt32> into a std::unordered_set ?? -namespace std -{ - template <> - struct hash< std::pair<RESOURCE_TYPE,sal_uInt32> > - { - size_t operator()(const std::pair<RESOURCE_TYPE,sal_uInt32> & x) const - { - return sal_uInt32(x.first) * 31 + x.second; - } - }; -} -class InternalResMgr -{ - friend class ResMgr; - friend class SimpleResMgr; - friend class ResMgrContainer; - - ImpContent * pContent; - sal_uInt32 nOffCorrection; - sal_uInt8 * pStringBlock; - SvStream * pStm; - bool bEqual2Content; - sal_uInt32 nEntries; - OUString aFileName; - OUString aPrefix; - OUString aResName; - bool bSingular; - LanguageTag aLocale; - std::unique_ptr<std::unordered_set<std::pair<RESOURCE_TYPE,sal_uInt32>>> pResUseDump; - - InternalResMgr( const OUString& rFileURL, - const OUString& aPrefix, - const OUString& aResName, - const LanguageTag& rLocale ); - bool Create(); - - bool IsGlobalAvailable( RESOURCE_TYPE nRT, sal_uInt32 nId ) const; - void * LoadGlobalRes( RESOURCE_TYPE nRT, sal_uInt32 nId, - void **pResHandle ); -public: - static void FreeGlobalRes( void const *, void * ); - ~InternalResMgr(); -}; - -class ResMgrContainer -{ - static ResMgrContainer* pOneInstance; - - struct ContainerElement - { - InternalResMgr* pResMgr; - OUString aFileURL; - int nRefCount; - int nLoadCount; - - ContainerElement() : - pResMgr( nullptr ), - nRefCount( 0 ), - nLoadCount( 0 ) - {} - }; - - std::unordered_map< OUString, ContainerElement, OUStringHash> m_aResFiles; - LanguageTag m_aDefLocale; - - ResMgrContainer() : m_aDefLocale( LANGUAGE_SYSTEM) { init(); } - ~ResMgrContainer(); - - void init(); - -public: - static ResMgrContainer& get(); - static void release(); - - InternalResMgr* getResMgr( const OUString& rPrefix, - LanguageTag& rLocale, - bool bForceNewInstance = false - ); - InternalResMgr* getNextFallback( InternalResMgr* pResMgr ); - - void freeResMgr( InternalResMgr* pResMgr ); - - void setDefLocale( const LanguageTag& rLocale ) - { m_aDefLocale = rLocale; } - const LanguageTag& getDefLocale() const - { return m_aDefLocale; } -}; - -ResMgrContainer* ResMgrContainer::pOneInstance = nullptr; - -ResMgrContainer& ResMgrContainer::get() -{ - if( ! pOneInstance ) - pOneInstance = new ResMgrContainer(); - return *pOneInstance; -} - -ResMgrContainer::~ResMgrContainer() -{ - for( std::pair< OUString, ContainerElement > const & rPair : m_aResFiles ) - { - SAL_INFO("tools.rc", "Resource file " << rPair.second.aFileURL << " loaded " << rPair.second.nLoadCount << " times"); - delete rPair.second.pResMgr; - } -} - -void ResMgrContainer::release() -{ - delete pOneInstance; - pOneInstance = nullptr; -} - -void ResMgrContainer::init() -{ - assert( m_aResFiles.empty() ); - - // get resource path - OUString uri("$BRAND_BASE_DIR/" LIBO_SHARE_RESOURCE_FOLDER "/"); - rtl::Bootstrap::expandMacros(uri); //TODO: detect failure - - // collect all possible resource files - Directory aDir( uri ); - if( aDir.open() == FileBase::E_None ) - { - DirectoryItem aItem; - while( aDir.getNextItem( aItem ) == FileBase::E_None ) - { - FileStatus aStatus(osl_FileStatus_Mask_FileName); - if( aItem.getFileStatus( aStatus ) == FileBase::E_None ) - { - OUString aFileName = aStatus.getFileName(); - if( ! aFileName.endsWithIgnoreAsciiCase( ".res" ) ) - continue; - OUString aResName = aFileName.copy( 0, aFileName.getLength() - strlen(".res") ); - if( aResName.isEmpty() ) - continue; - assert( m_aResFiles.find( aResName ) == m_aResFiles.end() ); - m_aResFiles[ aResName ].aFileURL = uri + aFileName; - SAL_INFO( - "tools.rc", - "ResMgrContainer: " << aResName << " -> " - << m_aResFiles[ aResName ].aFileURL ); - } - } - } - else - SAL_WARN( "tools.rc", "opening dir " << uri << " failed" ); - - // set default language - LanguageType nLang = MsLangId::getSystemUILanguage(); - m_aDefLocale.reset( nLang); -} - namespace { - bool isAlreadyPureenUS(const LanguageTag &rLocale) - { - return ( rLocale.getLanguageType() == LANGUAGE_ENGLISH_US ); - } -} - -InternalResMgr* ResMgrContainer::getResMgr( const OUString& rPrefix, - LanguageTag& rLocale, - bool bForceNewInstance - ) -{ - LanguageTag aLocale( rLocale ); - std::unordered_map< OUString, ContainerElement, OUStringHash >::iterator it = m_aResFiles.end(); - - ::std::vector< OUString > aFallbacks( aLocale.getFallbackStrings( true)); - if (!isAlreadyPureenUS( aLocale)) - aFallbacks.push_back( "en-US"); // last resort if all fallbacks fail - - for (::std::vector< OUString >::const_iterator fb( aFallbacks.begin()); fb != aFallbacks.end(); ++fb) - { - OUString aSearch( rPrefix + *fb ); - it = m_aResFiles.find( aSearch ); - if( it != m_aResFiles.end() ) - { - // ensure InternalResMgr existence - if( ! it->second.pResMgr ) - { - InternalResMgr* pImp = - new InternalResMgr( it->second.aFileURL, rPrefix, it->first, aLocale ); - if( ! pImp->Create() ) - { - delete pImp; - continue; - } - it->second.pResMgr = pImp; - } - break; - } - } - // try if there is anything with this prefix at all - if( it == m_aResFiles.end() ) - { - aLocale.reset( LANGUAGE_SYSTEM); - it = m_aResFiles.find( rPrefix ); - if( it == m_aResFiles.end() ) - { - for( it = m_aResFiles.begin(); it != m_aResFiles.end(); ++it ) - { - if( it->first.matchIgnoreAsciiCase( rPrefix ) ) - { - // ensure InternalResMgr existence - if( ! it->second.pResMgr ) - { - InternalResMgr* pImp = - new InternalResMgr( it->second.aFileURL, - rPrefix, - it->first, - aLocale ); - if( ! pImp->Create() ) - { - delete pImp; - continue; - } - it->second.pResMgr = pImp; - } - // try to guess locale - sal_Int32 nIndex = rPrefix.getLength(); - if (nIndex < it->first.getLength()) - aLocale.reset( it->first.copy( nIndex)); - else - { - SAL_WARN( "tools.rc", "ResMgrContainer::getResMgr: it->first " << - it->first << " shorter than prefix " << rPrefix); - } - break; - } - } - } - } - // give up - if( it == m_aResFiles.end() ) - { - OUString sURL = rPrefix + rLocale.getBcp47() + ".res"; - if ( m_aResFiles.find(sURL) == m_aResFiles.end() ) - { - m_aResFiles[ sURL ].aFileURL = sURL; - return getResMgr(rPrefix,rLocale,bForceNewInstance); - } // if ( m_aResFiles.find(sURL) == m_aResFiles.end() ) - return nullptr; - } - - rLocale = aLocale; - // at this point it->second.pResMgr must be filled either by creating a new one - // (then the refcount is still 0) or because we already had one - InternalResMgr* pImp = it->second.pResMgr; - - if( it->second.nRefCount == 0 ) - it->second.nLoadCount++; - - // for SimpleResMgr - if( bForceNewInstance ) - { - if( it->second.nRefCount == 0 ) - { - // shortcut: the match algorithm already created the InternalResMgr - // take it instead of creating yet another one - it->second.pResMgr = nullptr; - pImp->bSingular = true; - } - else - { - pImp = new InternalResMgr( it->second.aFileURL, rPrefix, it->first, aLocale ); - pImp->bSingular = true; - if( !pImp->Create() ) - { - delete pImp; - pImp = nullptr; - } - else - it->second.nLoadCount++; - } - } - else - it->second.nRefCount++; - - return pImp; -} - -InternalResMgr* ResMgrContainer::getNextFallback( InternalResMgr* pMgr ) -{ - /* TODO-BCP47: this is nasty, but the previous code simply stripped a - * locale's variant and country in subsequent calls to end up with language - * only and then fallback to en-US if all failed, so this is at least - * equivalent if not better. Maybe this method could be changed to get - * passed / remember a fallback list and an index within to pick the next. - * */ - - ::std::vector< OUString > aFallbacks( pMgr->aLocale.getFallbackStrings( true)); - // The first is the locale itself, use next fallback or en-US. - /* TODO: what happens if the chain is "en-US", "en" -> "en-US", ... - * This was already an issue with the previous code. */ - LanguageTag aLocale( ((aFallbacks.size() > 1) ? aFallbacks[1] : OUString( "en-US"))); - InternalResMgr* pNext = getResMgr( pMgr->aPrefix, aLocale, pMgr->bSingular ); - // prevent recursion - if( pNext == pMgr || ( pNext && pNext->aResName == pMgr->aResName ) ) - { - if( pNext->bSingular ) - delete pNext; - pNext = nullptr; - } - return pNext; -} - -void ResMgrContainer::freeResMgr( InternalResMgr* pResMgr ) -{ - if( pResMgr->bSingular ) - delete pResMgr; - else - { - std::unordered_map< OUString, ContainerElement, OUStringHash >::iterator it = - m_aResFiles.find( pResMgr->aResName ); - if( it != m_aResFiles.end() ) - { - DBG_ASSERT( it->second.nRefCount > 0, "InternalResMgr freed too often" ); - if( it->second.nRefCount > 0 ) - it->second.nRefCount--; - if( it->second.nRefCount == 0 ) - { - delete it->second.pResMgr; - it->second.pResMgr = nullptr; - } - } - } -} - -struct ImpContent -{ - RESOURCE_TYPE nType; - sal_uInt32 nId; - sal_uInt32 nOffset; -}; - -struct ImpContentLessCompare -{ - bool operator() (const ImpContent& rLhs, const ImpContent& rRhs) const + OUString createFromUtf8(const char* data, size_t size) { - sal_uInt64 const lhs((static_cast<sal_uInt64>(rLhs.nType.get()) << 32) | rLhs.nId); - sal_uInt64 const rhs((static_cast<sal_uInt64>(rRhs.nType.get()) << 32) | rRhs.nId); - return lhs < rhs; + OUString aTarget; + bool bSuccess = rtl_convertStringToUString(&aTarget.pData, + data, + size, + RTL_TEXTENCODING_UTF8, + RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_ERROR|RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_ERROR|RTL_TEXTTOUNICODE_FLAGS_INVALID_ERROR); + (void) bSuccess; + assert(bSuccess); + return aTarget; } -}; - -static ResHookProc pImplResHookProc = nullptr; - -InternalResMgr::InternalResMgr( const OUString& rFileURL, - const OUString& rPrefix, - const OUString& rResName, - const LanguageTag& rLocale ) - : pContent( nullptr ) - , nOffCorrection( 0 ) - , pStringBlock( nullptr ) - , pStm( nullptr ) - , bEqual2Content( true ) - , nEntries( 0 ) - , aFileName( rFileURL ) - , aPrefix( rPrefix ) - , aResName( rResName ) - , bSingular( false ) - , aLocale( rLocale ) -{ -} -InternalResMgr::~InternalResMgr() -{ - rtl_freeMemory(pContent); - rtl_freeMemory(pStringBlock); - delete pStm; - -#ifdef DBG_UTIL - if( pResUseDump ) + OString genKeyId(const OString& rGenerator) { - const sal_Char* pLogFile = getenv( "STAR_RESOURCE_LOGGING" ); - if ( pLogFile ) + sal_uInt32 nCRC = rtl_crc32(0, rGenerator.getStr(), rGenerator.getLength()); + // Use simple ASCII characters, exclude I, l, 1 and O, 0 to avoid confusing IDs + static const char sSymbols[] = + "ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz23456789"; + char sKeyId[6]; + for (short nKeyInd = 0; nKeyInd < 5; ++nKeyInd) { - SvFileStream aStm( OUString::createFromAscii( pLogFile ), StreamMode::WRITE ); - aStm.Seek( STREAM_SEEK_TO_END ); - OStringBuffer aLine("FileName: "); - aLine.append(OUStringToOString(aFileName, - RTL_TEXTENCODING_UTF8)); - aStm.WriteLine(aLine.makeStringAndClear()); - - for( auto const & rPair : *pResUseDump ) - { - aLine.append("Type/Id: "); - aLine.append((sal_Int64)sal_uInt32(rPair.first)); - aLine.append('/'); - aLine.append((sal_Int64)rPair.second); - aStm.WriteLine(aLine.makeStringAndClear()); - } + sKeyId[nKeyInd] = sSymbols[(nCRC & 63) % strlen(sSymbols)]; + nCRC >>= 6; } + sKeyId[5] = '\0'; + return OString(sKeyId); } -#endif } -bool InternalResMgr::Create() +namespace Translate { - ResMgrContainer::get(); - bool bDone = false; - - pStm = new SvFileStream( aFileName, StreamMode::READ | StreamMode::SHARE_DENYWRITE | StreamMode::NOCREATE ); - if( pStm->GetError() == ERRCODE_NONE ) + std::locale Create(const sal_Char* pPrefixName, const LanguageTag& rLocale) { - sal_Int32 lContLen = 0; - - pStm->Seek( STREAM_SEEK_TO_END ); - /* - if( ( pInternalResMgr->pHead = (RSHEADER_TYPE *)mmap( 0, nResourceFileSize, - PROT_READ, MAP_PRIVATE, - fRes, 0 ) ) != (RSHEADER_TYPE *)-1) - */ - pStm->SeekRel( - (int)sizeof( lContLen ) ); - pStm->ReadBytes( &lContLen, sizeof( lContLen ) ); - // file is bigendian but SvStreamEndian not set, swab to the right endian - lContLen = ResMgr::GetLong( &lContLen ); - pStm->SeekRel( -lContLen ); - // allocate stored ImpContent data (12 bytes per unit) - sal_uInt8* pContentBuf = static_cast<sal_uInt8*>(rtl_allocateMemory( lContLen )); - pStm->ReadBytes( pContentBuf, lContLen ); - // allocate ImpContent space (sizeof(ImpContent) per unit, not necessarily 12) - pContent = static_cast<ImpContent *>(rtl_allocateMemory( sizeof(ImpContent)*lContLen/12 )); - // Shorten to number of ImpContent - nEntries = (sal_uInt32)lContLen / 12; - bEqual2Content = true; - bool bSorted = true; - if( nEntries ) - { -#ifdef DBG_UTIL - const sal_Char* pLogFile = getenv( "STAR_RESOURCE_LOGGING" ); - if ( pLogFile ) - { - pResUseDump.reset( new std::unordered_set<std::pair<RESOURCE_TYPE, sal_uInt32>> ); - for( sal_uInt32 i = 0; i < nEntries; ++i ) - pResUseDump->insert({pContent[i].nType, pContent[i].nId}); - } -#endif - // swap the content to the right endian - pContent[0].nType = RESOURCE_TYPE(ResMgr::GetLong( pContentBuf )); - pContent[0].nId = ResMgr::GetLong( pContentBuf + 4); - pContent[0].nOffset = ResMgr::GetLong( pContentBuf + 8 ); - sal_uInt32 nCount = nEntries - 1; - for( sal_uInt32 i = 0,j=1; i < nCount; ++i,++j ) - { - // swap the content to the right endian - pContent[j].nType = RESOURCE_TYPE(ResMgr::GetLong( pContentBuf + (12*j) )); - pContent[j].nId = ResMgr::GetLong( pContentBuf + (12*j) + 4 ); - pContent[j].nOffset = ResMgr::GetLong( pContentBuf + (12*j) + 8 ); - if( std::tie(pContent[i].nType, pContent[i].nId) >= std::tie(pContent[j].nType, pContent[j].nId) ) - bSorted = false; - if( pContent[i].nId == pContent[j].nId - && pContent[i].nOffset >= pContent[j].nOffset ) - bEqual2Content = false; - } - } - rtl_freeMemory( pContentBuf ); - OSL_ENSURE( bSorted, "content not sorted" ); - OSL_ENSURE( bEqual2Content, "resource structure wrong" ); - if( !bSorted ) - ::std::sort(pContent,pContent+nEntries,ImpContentLessCompare()); - // qsort( pContent, nEntries, sizeof( ImpContent ), Compare ); - - bDone = true; + boost::locale::generator gen; + gen.characters(boost::locale::char_facet); + gen.categories(boost::locale::message_facet | boost::locale::information_facet); + OUString uri("$BRAND_BASE_DIR/$BRAND_SHARE_RESOURCE_SUBDIR/"); + rtl::Bootstrap::expandMacros(uri); + OUString path; + osl::File::getSystemPathFromFileURL(uri, path); + gen.add_messages_path(OUStringToOString(path, osl_getThreadTextEncoding()).getStr()); + gen.add_messages_domain(pPrefixName); + OString sIdentifier = rLocale.getGlibcLocaleString(".UTF-8").toUtf8(); + return gen(sIdentifier.getStr()); } - return bDone; -} - - -bool InternalResMgr::IsGlobalAvailable( RESOURCE_TYPE nRT, sal_uInt32 nId ) const -{ - // search beginning of string - ImpContent aValue; - aValue.nType = nRT; - aValue.nId = nId; - ImpContent * pFind = ::std::lower_bound(pContent, - pContent + nEntries, - aValue, - ImpContentLessCompare()); - return (pFind != (pContent + nEntries)) && (pFind->nType == aValue.nType) && (pFind->nId == aValue.nId); -} - - -void* InternalResMgr::LoadGlobalRes( RESOURCE_TYPE nRT, sal_uInt32 nId, - void **pResHandle ) -{ -#ifdef DBG_UTIL - if( pResUseDump ) - pResUseDump->erase( { nRT, nId } ); -#endif - // search beginning of string - ImpContent aValue; - aValue.nType = nRT; - aValue.nId = nId; - ImpContent* pEnd = (pContent + nEntries); - ImpContent* pFind = ::std::lower_bound( pContent, - pEnd, - aValue, - ImpContentLessCompare()); - if( pFind && (pFind != pEnd) && (pFind->nType == aValue.nType) && (pFind->nId == aValue.nId) ) + OUString get(const char* pContextAndId, const std::locale &loc) { - if( nRT == RSC_STRING && bEqual2Content ) - { - // string optimization - if( !pStringBlock ) - { - // search beginning of string - ImpContent * pFirst = pFind; - ImpContent * pLast = pFirst; - while( pFirst > pContent && (pFirst -1)->nType == RSC_STRING ) - --pFirst; - while( pLast < pEnd && pLast->nType == RSC_STRING ) - ++pLast; - nOffCorrection = pFirst->nOffset; - sal_uInt32 nSize; - --pLast; - pStm->Seek( pLast->nOffset ); - RSHEADER_TYPE aHdr; - pStm->ReadBytes( &aHdr, sizeof( aHdr ) ); - nSize = pLast->nOffset + aHdr.GetGlobOff() - nOffCorrection; - pStringBlock = static_cast<sal_uInt8*>(rtl_allocateMemory( nSize )); - pStm->Seek( pFirst->nOffset ); - pStm->ReadBytes( pStringBlock, nSize ); - } - *pResHandle = pStringBlock; - return pStringBlock + pFind->nOffset - nOffCorrection; - } // if( nRT == RSC_STRING && bEqual2Content ) + OString sContext; + const char *pId = strchr(pContextAndId, '\004'); + if (!pId) + pId = pContextAndId; else { - *pResHandle = nullptr; - RSHEADER_TYPE aHeader; - pStm->Seek( pFind->nOffset ); - pStm->ReadBytes( &aHeader, sizeof( RSHEADER_TYPE ) ); - void * pRes = rtl_allocateMemory( aHeader.GetGlobOff() ); - memcpy( pRes, &aHeader, sizeof( RSHEADER_TYPE ) ); - pStm->ReadBytes(static_cast<sal_uInt8*>(pRes) + sizeof(RSHEADER_TYPE), - aHeader.GetGlobOff() - sizeof( RSHEADER_TYPE ) ); - return pRes; + sContext = OString(pContextAndId, pId - pContextAndId); + ++pId; } - } // if( pFind && (pFind != pEnd) && (pFind->nTypeAndId == nValue) ) - *pResHandle = nullptr; - return nullptr; -} - -void InternalResMgr::FreeGlobalRes( void const * pResHandle, void * pResource ) -{ - if ( !pResHandle ) - // Free allocated resource - rtl_freeMemory(pResource); -} - -#ifdef DBG_UTIL -OUString GetTypeRes_Impl( const ResId& rTypeId ) -{ - // Return on resource errors - static bool bInUse = false; - OUString aTypStr(OUString::number(rTypeId.GetId())); - - if ( !bInUse ) - { - bInUse = true; - - ResId aResId( sal_uInt32(RSCVERSION_ID), *rTypeId.GetResMgr() ); - aResId.SetRT( RSC_VERSIONCONTROL ); - - if ( rTypeId.GetResMgr()->GetResource( aResId ) ) + //if its a key id locale, generate it here + if (std::use_facet<boost::locale::info>(loc).language() == "qtz") { - rTypeId.SetRT( RSC_STRING ); - if ( rTypeId.GetResMgr()->IsAvailable( rTypeId ) ) - { - aTypStr = rTypeId; - // Set class pointer to the end - rTypeId.GetResMgr()->Increment( sizeof( RSHEADER_TYPE ) ); - } + OString sKeyId(genKeyId(OString(pContextAndId).replace('\004', '|'))); + return OUString::fromUtf8(sKeyId) + OUStringLiteral1(0x2016) + createFromUtf8(pId, strlen(pId)); } - bInUse = false; - } - - return aTypStr; -} - -void ResMgr::RscError_Impl( const sal_Char* pMessage, ResMgr* pResMgr, - RESOURCE_TYPE nRT, sal_uInt32 nId, - std::vector< ImpRCStack >& rResStack, int nDepth ) -{ - // create a separate ResMgr with its own stack - // first get a second reference of the InternalResMgr - InternalResMgr* pImp = - ResMgrContainer::get().getResMgr( pResMgr->pImpRes->aPrefix, - pResMgr->pImpRes->aLocale, - true ); - - ResMgr* pNewResMgr = new ResMgr( pImp ); - - OString aFilename(OUStringToOString(pResMgr->GetFileName(), - RTL_TEXTENCODING_UTF8)); - OStringBuffer aStr(pMessage); - aStr.append(aFilename); - aStr.append('\n'); - aStr.append(" Class: "); - aStr.append(OUStringToOString(GetTypeRes_Impl(ResId((sal_uInt32)nRT, *pNewResMgr)), - RTL_TEXTENCODING_UTF8)); - aStr.append(", Id: "); - aStr.append(static_cast<sal_Int32>(nId)); - aStr.append(". "); - - aStr.append(" Resource Stack:"); - while( nDepth > 0 ) - { - aStr.append(" [ Class: "); - aStr.append(OUStringToOString(GetTypeRes_Impl( - ResId((sal_uInt32)rResStack[nDepth].pResource->GetRT(), *pNewResMgr)), - RTL_TEXTENCODING_UTF8)); - aStr.append(", Id: "); - aStr.append(static_cast<sal_Int32>( - rResStack[nDepth].pResource->GetId())); - aStr.append("]"); - nDepth--; + //otherwise translate it + const std::string ret = boost::locale::pgettext(sContext.getStr(), pId, loc); + return ExpandVariables(createFromUtf8(ret.data(), ret.size())); } - // clean up - delete pNewResMgr; - - OSL_FAIL(aStr.getStr()); -} - -#endif + static ResHookProc pImplResHookProc = nullptr; -static void RscException_Impl() -{ - switch ( osl_raiseSignal( OSL_SIGNAL_USER_RESOURCEFAILURE, const_cast<char *>("") ) ) + OUString ExpandVariables(const OUString& rString) { - case osl_Signal_ActCallNextHdl: - abort(); - - case osl_Signal_ActIgnore: - return; - - case osl_Signal_ActAbortApp: - abort(); - - default: - case osl_Signal_ActKillApp: - exit(-1); - } -} - -void ImpRCStack::Init( const Resource* pObj, sal_uInt32 Id ) -{ - pResource = nullptr; - pClassRes = nullptr; - Flags = RCFlags::NONE; - aResHandle = nullptr; - pResObj = pObj; - if ( !(Id & RSC_DONTRELEASE) ) - Flags |= RCFlags::AUTORELEASE; -} - -void ImpRCStack::Clear() -{ - pResource = nullptr; - pClassRes = nullptr; - Flags = RCFlags::NONE; - aResHandle = nullptr; - pResObj = nullptr; -} - -static RSHEADER_TYPE* LocalResource( const ImpRCStack* pStack, - RESOURCE_TYPE nRTType, - sal_uInt32 nId ) -{ - // Returns position of the resource if found or NULL otherwise - - if ( pStack->pResource && pStack->pClassRes ) - { - RSHEADER_TYPE* pTmp; // Pointer to child resource - RSHEADER_TYPE* pEnd; // Pointer to the end of this resource - pTmp = reinterpret_cast<RSHEADER_TYPE*> - (reinterpret_cast<sal_uInt8*>(pStack->pResource) + pStack->pResource->GetLocalOff()); - pEnd = reinterpret_cast<RSHEADER_TYPE*> - (reinterpret_cast<sal_uInt8*>(pStack->pResource) + pStack->pResource->GetGlobOff()); - while ( pTmp != pEnd ) - { - if ( pTmp->GetRT() == nRTType && pTmp->GetId() == nId ) - return pTmp; - pTmp = reinterpret_cast<RSHEADER_TYPE*>(reinterpret_cast<sal_uInt8*>(pTmp) + pTmp->GetGlobOff()); - } + if (pImplResHookProc) + return pImplResHookProc(rString); + return rString; } - return nullptr; -} - -void* ResMgr::pEmptyBuffer = nullptr; - -void* ResMgr::getEmptyBuffer() -{ - if( ! pEmptyBuffer ) - pEmptyBuffer = rtl_allocateZeroMemory( 1024 ); - return pEmptyBuffer; -} - -void ResMgr::DestroyAllResMgr() -{ + void SetReadStringHook( ResHookProc pProc ) { - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - if( pEmptyBuffer ) - { - rtl_freeMemory( pEmptyBuffer ); - pEmptyBuffer = nullptr; - } - ResMgrContainer::release(); + pImplResHookProc = pProc; } - delete pResMgrMutex; - pResMgrMutex = nullptr; -} - -void ResMgr::Init( const OUString& rFileName ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - if ( !pImpRes ) + ResHookProc GetReadStringHook() { - SAL_WARN( "tools.rc", "Resourcefile not found: " << rFileName); - RscException_Impl(); - } -#ifdef DBG_UTIL - else - { - void* aResHandle = nullptr; // Helper variable for resource handles - void* pVoid; // Pointer on the resource - - pVoid = pImpRes->LoadGlobalRes( RSC_VERSIONCONTROL, RSCVERSION_ID, - &aResHandle ); - if ( pVoid ) - InternalResMgr::FreeGlobalRes( aResHandle, pVoid ); - else - { - SAL_WARN("tools.rc", "Wrong version: " << pImpRes->aFileName); - } - } -#endif - nCurStack = -1; - aStack.clear(); - pFallbackResMgr = pOriginalResMgr = nullptr; - incStack(); -} - -ResMgr::ResMgr( InternalResMgr * pImpMgr ) -{ - pImpRes = pImpMgr; - Init( pImpMgr->aFileName ); -} - -ResMgr::~ResMgr() -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - ResMgrContainer::get().freeResMgr( pImpRes ); - - // clean up possible left rc stack frames - while( nCurStack > 0 ) - { - if( ( aStack[nCurStack].Flags & (RCFlags::GLOBAL | RCFlags::NOTFOUND) ) == RCFlags::GLOBAL ) - InternalResMgr::FreeGlobalRes( aStack[nCurStack].aResHandle, - aStack[nCurStack].pResource ); - nCurStack--; - } -} - -void ResMgr::incStack() -{ - nCurStack++; - if( nCurStack >= int(aStack.size()) ) - aStack.push_back( ImpRCStack() ); - aStack[nCurStack].Clear(); - - DBG_ASSERT( nCurStack < 32, "Resource stack unreasonably large" ); -} - -void ResMgr::decStack() -{ - DBG_ASSERT( nCurStack > 0, "resource stack underrun !" ); - if( (aStack[nCurStack].Flags & RCFlags::FALLBACK_UP) ) - { - nCurStack--; - // warning: this will delete *this, see below - pOriginalResMgr->decStack(); - } - else - { - ImpRCStack& rTop = aStack[nCurStack]; - if( (rTop.Flags & RCFlags::FALLBACK_DOWN) ) - { - #if OSL_DEBUG_LEVEL > 1 - SAL_INFO("tools", "returning from fallback " << pFallbackResMgr->GetFileName() ); - #endif - delete pFallbackResMgr; - pFallbackResMgr = nullptr; - } - nCurStack--; - } -} - -bool ResMgr::IsAvailable( const ResId& rId, const Resource* pResObj ) const -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - bool bAvailable = false; - RSHEADER_TYPE* pClassRes = rId.GetpResource(); - RESOURCE_TYPE nRT = rId.GetRT(); - sal_uInt32 nId = rId.GetId(); - const ResMgr* pMgr = rId.GetResMgr(); - - if ( !pMgr ) - pMgr = this; - - if( pMgr->pFallbackResMgr ) - { - ResId aId( rId ); - aId.ClearResMgr(); - return pMgr->pFallbackResMgr->IsAvailable( aId, pResObj ); - } - - if ( !pResObj || pResObj == pMgr->aStack[pMgr->nCurStack].pResObj ) - { - if ( !pClassRes ) - pClassRes = LocalResource( &pMgr->aStack[pMgr->nCurStack], nRT, nId ); - if ( pClassRes ) - { - if ( pClassRes->GetRT() == nRT ) - bAvailable = true; - } - } - - if ( !pClassRes ) - bAvailable = pMgr->pImpRes->IsGlobalAvailable( nRT, nId ); - - return bAvailable; -} - -void* ResMgr::GetClass() -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - if( pFallbackResMgr ) - return pFallbackResMgr->GetClass(); - - return aStack[nCurStack].pClassRes; -} - -bool ResMgr::GetResource( const ResId& rId, const Resource* pResObj ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - if( pFallbackResMgr ) - { - ResId aId( rId ); - aId.ClearResMgr(); - return pFallbackResMgr->GetResource( aId, pResObj ); - } - - ResMgr* pMgr = rId.GetResMgr(); - if ( pMgr && (this != pMgr) ) - return pMgr->GetResource( rId, pResObj ); - - // normally Increment will pop the context; this is - // not possible in RCFlags::NOTFOUND case, so pop a frame here - ImpRCStack* pTop = &aStack[nCurStack]; - if( (pTop->Flags & RCFlags::NOTFOUND) ) - { - decStack(); - } - - RSHEADER_TYPE* pClassRes = rId.GetpResource(); - RESOURCE_TYPE nRT = rId.GetRT(); - sal_uInt32 nId = rId.GetId(); - - incStack(); - pTop = &aStack[nCurStack]; - pTop->Init( pResObj, nId | - (rId.IsAutoRelease() ? 0 : RSC_DONTRELEASE) ); - - if ( pClassRes ) - { - if ( pClassRes->GetRT() == nRT ) - pTop->pClassRes = pClassRes; - else - { -#ifdef DBG_UTIL - RscError_Impl( "Different class and resource type!", - this, nRT, nId, aStack, nCurStack-1 ); -#endif - pTop->Flags |= RCFlags::NOTFOUND; - pTop->pClassRes = getEmptyBuffer(); - pTop->pResource = static_cast<RSHEADER_TYPE*>(pTop->pClassRes); - return false; - } - } - else - { - OSL_ENSURE( nCurStack > 0, "stack of 1 to shallow" ); - pTop->pClassRes = LocalResource( &aStack[nCurStack-1], nRT, nId ); - } - - if ( pTop->pClassRes ) - // local Resource, not a system Resource - pTop->pResource = static_cast<RSHEADER_TYPE *>(pTop->pClassRes); - else - { - pTop->pClassRes = pImpRes->LoadGlobalRes( nRT, nId, &pTop->aResHandle ); - if ( pTop->pClassRes ) - { - pTop->Flags |= RCFlags::GLOBAL; - pTop->pResource = static_cast<RSHEADER_TYPE *>(pTop->pClassRes); - } - else - { - // try to get a fallback resource - pFallbackResMgr = CreateFallbackResMgr( rId, pResObj ); - if( pFallbackResMgr ) - { - pTop->Flags |= RCFlags::FALLBACK_DOWN; -#ifdef DBG_UTIL - OStringBuffer aMess("found resource "); - aMess.append(static_cast<sal_Int32>(nId)); - aMess.append(" in fallback "); - aMess.append(OUStringToOString( - pFallbackResMgr->GetFileName(), - osl_getThreadTextEncoding())); - aMess.append('\n'); - RscError_Impl(aMess.getStr(), - this, nRT, nId, aStack, nCurStack-1); -#endif - } - else - { - #ifdef DBG_UTIL - RscError_Impl( "Cannot load resource! ", - this, nRT, nId, aStack, nCurStack-1 ); - #endif - pTop->Flags |= RCFlags::NOTFOUND; - pTop->pClassRes = getEmptyBuffer(); - pTop->pResource = static_cast<RSHEADER_TYPE*>(pTop->pClassRes); - return false; - } - } - } - - return true; -} - -void ResMgr::PopContext( const Resource* pResObj ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - if( pFallbackResMgr ) - { - pFallbackResMgr->PopContext( pResObj ); - return; - } - - if ( nCurStack > 0 ) - { - ImpRCStack* pTop = &aStack[nCurStack]; - // free resource - if( (pTop->Flags & (RCFlags::GLOBAL | RCFlags::NOTFOUND)) == RCFlags::GLOBAL ) - // free global resource if resource is foreign - InternalResMgr::FreeGlobalRes( pTop->aResHandle, pTop->pResource ); - decStack(); - } -} - -sal_Int32 ResMgr::GetLong( void const * pLong ) -{ - return ((*(static_cast<const sal_uInt8*>(pLong) + 0) << 24) | - (*(static_cast<const sal_uInt8*>(pLong) + 1) << 16) | - (*(static_cast<const sal_uInt8*>(pLong) + 2) << 8) | - (*(static_cast<const sal_uInt8*>(pLong) + 3) << 0) ); -} - -sal_uInt32 ResMgr::GetStringWithoutHook( OUString& rStr, const sal_uInt8* pStr ) -{ - sal_uInt32 nLen=0; - sal_uInt32 nRet = GetStringSize( pStr, nLen ); - const sal_Char* str = reinterpret_cast< const sal_Char* >( pStr ); - OUString aString( str, strlen( str ), RTL_TEXTENCODING_UTF8, - RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_MAPTOPRIVATE | - RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_DEFAULT | - RTL_TEXTTOUNICODE_FLAGS_INVALID_DEFAULT ); - rStr = aString; - return nRet; -} - -sal_uInt32 ResMgr::GetString( OUString& rStr, const sal_uInt8* pStr ) -{ - OUString aString; - sal_uInt32 nRet = GetStringWithoutHook( aString, pStr ); - if ( pImplResHookProc ) - aString = pImplResHookProc( aString ); - rStr = aString; - return nRet; -} - -sal_uInt32 ResMgr::GetStringSize( const sal_uInt8* pStr, sal_uInt32& nLen ) -{ - nLen = static_cast< sal_uInt32 >( strlen( reinterpret_cast<const char*>(pStr) ) ); - return GetStringSize( nLen ); -} - -void* ResMgr::Increment( sal_uInt32 nSize ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - if( pFallbackResMgr ) - return pFallbackResMgr->Increment( nSize ); - - ImpRCStack& rStack = aStack[nCurStack]; - if( (rStack.Flags & RCFlags::NOTFOUND) ) - return rStack.pClassRes; - - sal_uInt8* pClassRes = static_cast<sal_uInt8*>(rStack.pClassRes) + nSize; - - rStack.pClassRes = pClassRes; - - RSHEADER_TYPE* pRes = rStack.pResource; - - sal_uInt32 nLocalOff = pRes->GetLocalOff(); - if ( (pRes->GetGlobOff() == nLocalOff) && - ((reinterpret_cast<char*>(pRes) + nLocalOff) == rStack.pClassRes) && - (rStack.Flags & RCFlags::AUTORELEASE)) - { - PopContext( rStack.pResObj ); - } - - return pClassRes; -} - -ResMgr* ResMgr::CreateFallbackResMgr( const ResId& rId, const Resource* pResource ) -{ - ResMgr *pFallback = nullptr; - if( nCurStack > 0 ) - { - // get the next fallback level in resource file scope - InternalResMgr* pRes = ResMgrContainer::get().getNextFallback( pImpRes ); - if( pRes ) - { - // check that the fallback locale is not already in the chain of - // fallbacks - prevent fallback loops - ResMgr* pResMgr = this; - while( pResMgr && (pResMgr->pImpRes->aLocale != pRes->aLocale)) - { - pResMgr = pResMgr->pOriginalResMgr; - } - if( pResMgr ) - { - // found a recursion, no fallback possible - ResMgrContainer::get().freeResMgr( pRes ); - return nullptr; - } - SAL_INFO("tools.rc", "trying fallback: " << pRes->aFileName ); - pFallback = new ResMgr( pRes ); - pFallback->pOriginalResMgr = this; - // try to recreate the resource stack - bool bHaveStack = true; - for( int i = 1; i < nCurStack; i++ ) - { - if( !aStack[i].pResource ) - { - bHaveStack = false; - break; - } - ResId aId( aStack[i].pResource->GetId(), *pFallbackResMgr ); - aId.SetRT( aStack[i].pResource->GetRT() ); - if( !pFallback->GetResource( aId ) ) - { - bHaveStack = false; - break; - } - } - if( bHaveStack ) - { - ResId aId( rId.GetId(), *pFallback ); - aId.SetRT( rId.GetRT() ); - if( !pFallback->GetResource( aId, pResource ) ) - bHaveStack = false; - else - pFallback->aStack[pFallback->nCurStack].Flags |= RCFlags::FALLBACK_UP; - } - if( !bHaveStack ) - { - delete pFallback; - pFallback = nullptr; - } - } - } - return pFallback; -} - -ResMgr* ResMgr::CreateResMgr( const sal_Char* pPrefixName, - const LanguageTag& _aLocale ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - OUString aPrefix( pPrefixName, strlen( pPrefixName ), osl_getThreadTextEncoding() ); - - LanguageTag aLocale = _aLocale; - if( aLocale.isSystemLocale() ) - aLocale = ResMgrContainer::get().getDefLocale(); - - InternalResMgr* pImp = ResMgrContainer::get().getResMgr( aPrefix, aLocale ); - return pImp ? new ResMgr( pImp ) : nullptr; -} - -ResMgr* ResMgr::SearchCreateResMgr( - const sal_Char* pPrefixName, - LanguageTag& rLocale ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - OUString aPrefix( pPrefixName, strlen( pPrefixName ), osl_getThreadTextEncoding() ); - - if( rLocale.isSystemLocale() ) - rLocale = ResMgrContainer::get().getDefLocale(); - - InternalResMgr* pImp = ResMgrContainer::get().getResMgr( aPrefix, rLocale ); - return pImp ? new ResMgr( pImp ) : nullptr; -} - -sal_Int32 ResMgr::ReadLong() -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - if( pFallbackResMgr ) - return pFallbackResMgr->ReadLong(); - - sal_Int32 n = GetLong( GetClass() ); - Increment( sizeof( sal_Int32 ) ); - return n; -} - -OUString ResMgr::ReadStringWithoutHook() -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - - if( pFallbackResMgr ) - return pFallbackResMgr->ReadStringWithoutHook(); - - OUString aRet; - - const ImpRCStack& rTop = aStack[nCurStack]; - if( (rTop.Flags & RCFlags::NOTFOUND) ) - { - #if OSL_DEBUG_LEVEL > 0 - aRet = "<resource not found>"; - #endif - } - else - Increment( GetStringWithoutHook( aRet, static_cast<const sal_uInt8*>(GetClass()) ) ); - - return aRet; -} - -OUString ResMgr::ExpandVariables(const OUString& rString) -{ - if (pImplResHookProc) - return pImplResHookProc(rString); - return rString; -} - -OUString ResMgr::ReadString() -{ - return ExpandVariables(ReadStringWithoutHook()); -} - -void ResMgr::SetReadStringHook( ResHookProc pProc ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - pImplResHookProc = pProc; -} - -ResHookProc ResMgr::GetReadStringHook() -{ - return pImplResHookProc; -} - -void ResMgr::SetDefaultLocale( const LanguageTag& rLocale ) -{ - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - ResMgrContainer::get().setDefLocale( rLocale ); -} - -const OUString& ResMgr::GetFileName() const -{ - return pImpRes->aFileName; -} - -SimpleResMgr::SimpleResMgr( const sal_Char* pPrefixName, - const LanguageTag& rLocale ) -{ - OUString aPrefix( pPrefixName, strlen( pPrefixName ), osl_getThreadTextEncoding() ); - LanguageTag aLocale( rLocale ); - - osl::Guard<osl::Mutex> aGuard( getResMgrMutex() ); - if( aLocale.isSystemLocale() ) - aLocale = ResMgrContainer::get().getDefLocale(); - - m_pResImpl.reset(ResMgrContainer::get().getResMgr( aPrefix, aLocale, true )); - DBG_ASSERT( m_pResImpl, "SimpleResMgr::SimpleResMgr : have no impl class !" ); -} - -SimpleResMgr::~SimpleResMgr() -{ -} - -SimpleResMgr* SimpleResMgr::Create(const sal_Char* pPrefixName, const LanguageTag& rLocale) -{ - return new SimpleResMgr(pPrefixName, rLocale); -} - -bool SimpleResMgr::IsAvailable( RESOURCE_TYPE _resourceType, sal_uInt32 _resourceId ) -{ - osl::MutexGuard aGuard(m_aAccessSafety); - - if ( ( RSC_STRING != _resourceType ) && ( RSC_RESOURCE != _resourceType ) ) - return false; - - DBG_ASSERT( m_pResImpl, "SimpleResMgr::IsAvailable: have no impl class !" ); - return m_pResImpl->IsGlobalAvailable( _resourceType, _resourceId ); -} - -OUString SimpleResMgr::ReadString( sal_uInt32 nId ) -{ - osl::MutexGuard aGuard(m_aAccessSafety); - - DBG_ASSERT( m_pResImpl, "SimpleResMgr::ReadString : have no impl class !" ); - // perhaps constructed with an invalid filename ? - - OUString sReturn; - if ( !m_pResImpl ) - return sReturn; - - void* pResHandle = nullptr; - InternalResMgr* pFallback = m_pResImpl.get(); - RSHEADER_TYPE* pResHeader = static_cast<RSHEADER_TYPE*>(m_pResImpl->LoadGlobalRes( RSC_STRING, nId, &pResHandle )); - if ( !pResHeader ) - { - osl::Guard<osl::Mutex> aGuard2( getResMgrMutex() ); - - // try fallback - while( ! pResHandle && pFallback ) - { - InternalResMgr* pOldFallback = pFallback; - pFallback = ResMgrContainer::get().getNextFallback( pFallback ); - if( pOldFallback != m_pResImpl.get() ) - ResMgrContainer::get().freeResMgr( pOldFallback ); - if( pFallback ) - { - // handle possible recursion - if( pFallback->aLocale != m_pResImpl->aLocale ) - { - pResHeader = static_cast<RSHEADER_TYPE*>(pFallback->LoadGlobalRes( RSC_STRING, nId, &pResHandle )); - } - else - { - ResMgrContainer::get().freeResMgr( pFallback ); - pFallback = nullptr; - } - } - } - if( ! pResHandle ) - // no such resource - return sReturn; - } - - // sal_uIntPtr nLen = pResHeader->GetLocalOff() - sizeof(RSHEADER_TYPE); - ResMgr::GetString( sReturn, reinterpret_cast<sal_uInt8*>(pResHeader+1) ); - - // not necessary with the current implementation which holds the string table permanently, but to be sure .... - // note: pFallback cannot be NULL here and is either the fallback or m_pResImpl - InternalResMgr::FreeGlobalRes( pResHeader, pResHandle ); - if( m_pResImpl.get() != pFallback ) - { - osl::Guard<osl::Mutex> aGuard2( getResMgrMutex() ); - - ResMgrContainer::get().freeResMgr( pFallback ); + return pImplResHookProc; } - return sReturn; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */ |