blob: 9afe4f5480cf8e01a6cf1f0b102da54cc6508f3d [file] [log] [blame]
/**************************************************************
*
* 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
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_extensions.hxx"
#ifndef EXTENSIONS_SOURCE_RESOURCE_OOORESOURCELOADER_CXX
#define EXTENSIONS_SOURCE_RESOURCE_OOORESOURCELOADER_CXX
#include "res_services.hxx"
/** === begin UNO includes === **/
#include <com/sun/star/resource/XResourceBundleLoader.hpp>
#include <com/sun/star/lang/XMultiServiceFactory.hpp>
#include <com/sun/star/uno/XComponentContext.hpp>
/** === end UNO includes === **/
#include <vcl/svapp.hxx>
#include <tools/simplerm.hxx>
#include <tools/rcid.h>
#include <cppuhelper/implbase1.hxx>
#include <cppuhelper/weakref.hxx>
#include <boost/shared_ptr.hpp>
#include <map>
//........................................................................
namespace res
{
//........................................................................
/** === begin UNO using === **/
using ::com::sun::star::uno::Reference;
using ::com::sun::star::resource::XResourceBundleLoader;
using ::com::sun::star::resource::XResourceBundle;
using ::com::sun::star::resource::MissingResourceException;
using ::com::sun::star::uno::XComponentContext;
using ::com::sun::star::uno::Sequence;
using ::com::sun::star::uno::XInterface;
using ::com::sun::star::uno::RuntimeException;
using ::com::sun::star::lang::Locale;
using ::com::sun::star::uno::Any;
using ::com::sun::star::container::NoSuchElementException;
using ::com::sun::star::lang::WrappedTargetException;
using ::com::sun::star::uno::Type;
using ::com::sun::star::uno::WeakReference;
/** === end UNO using === **/
//====================================================================
//= helper
//====================================================================
typedef ::std::pair< ::rtl::OUString, Locale > ResourceBundleDescriptor;
struct ResourceBundleDescriptorLess : public ::std::binary_function< ResourceBundleDescriptor, ResourceBundleDescriptor, bool >
{
bool operator()( const ResourceBundleDescriptor& _lhs, const ResourceBundleDescriptor& _rhs ) const
{
if ( _lhs.first < _rhs.first )
return true;
if ( _lhs.second.Language < _rhs.second.Language )
return true;
if ( _lhs.second.Country < _rhs.second.Country )
return true;
if ( _lhs.second.Variant < _rhs.second.Variant )
return true;
return false;
}
};
//====================================================================
//= OpenOfficeResourceLoader
//====================================================================
typedef ::cppu::WeakImplHelper1 < XResourceBundleLoader
> OpenOfficeResourceLoader_Base;
class OpenOfficeResourceLoader : public OpenOfficeResourceLoader_Base
{
private:
typedef ::std::map< ResourceBundleDescriptor, WeakReference< XResourceBundle >, ResourceBundleDescriptorLess >
ResourceBundleCache;
private:
Reference< XComponentContext > m_xContext;
::osl::Mutex m_aMutex;
ResourceBundleCache m_aBundleCache;
protected:
OpenOfficeResourceLoader( const Reference< XComponentContext >& _rxContext );
public:
static Sequence< ::rtl::OUString > getSupportedServiceNames_static();
static ::rtl::OUString getImplementationName_static();
static ::rtl::OUString getSingletonName_static();
static Reference< XInterface > Create( const Reference< XComponentContext >& _rxContext );
// XResourceBundleLoader
virtual Reference< XResourceBundle > SAL_CALL loadBundle_Default( const ::rtl::OUString& aBaseName ) throw (MissingResourceException, RuntimeException);
virtual Reference< XResourceBundle > SAL_CALL loadBundle( const ::rtl::OUString& abaseName, const Locale& aLocale ) throw (MissingResourceException, RuntimeException);
private:
OpenOfficeResourceLoader(); // never implemented
OpenOfficeResourceLoader( const OpenOfficeResourceLoader& ); // never implemented
OpenOfficeResourceLoader& operator=( const OpenOfficeResourceLoader& ); // never implemented
};
//====================================================================
//= IResourceType
//====================================================================
/** encapsulates access to a fixed resource type
*/
class IResourceType
{
public:
/** returns the RESOURCE_TYPE associated with this instance
*/
virtual RESOURCE_TYPE getResourceType() const = 0;
/** reads a single resource from the given resource manager
@param _resourceManager
the resource manager to read from
@param _resourceId
the id of the resource to read
@return
the required resource
@precond
the caler checked via <code>_resourceManager.IsAvailable( getResourceType(), _resourceId )</code>
that the required resource really exists
*/
virtual Any getResource( SimpleResMgr& _resourceManager, sal_Int32 _resourceId ) const = 0;
virtual ~IResourceType() { };
};
//====================================================================
//= StringResourceAccess
//====================================================================
class StringResourceAccess : public IResourceType
{
public:
StringResourceAccess();
// IResourceType
virtual RESOURCE_TYPE getResourceType() const;
virtual Any getResource( SimpleResMgr& _resourceManager, sal_Int32 _resourceId ) const;
};
//--------------------------------------------------------------------
StringResourceAccess::StringResourceAccess()
{
}
//--------------------------------------------------------------------
RESOURCE_TYPE StringResourceAccess::getResourceType() const
{
return RSC_STRING;
}
//--------------------------------------------------------------------
Any StringResourceAccess::getResource( SimpleResMgr& _resourceManager, sal_Int32 _resourceId ) const
{
OSL_PRECOND( _resourceManager.IsAvailable( getResourceType(), _resourceId ), "StringResourceAccess::getResource: precondition not met!" );
Any aResource;
aResource <<= ::rtl::OUString( _resourceManager.ReadString( _resourceId ) );
return aResource;
}
//====================================================================
//= OpenOfficeResourceBundle
//====================================================================
typedef ::cppu::WeakImplHelper1 < XResourceBundle
> OpenOfficeResourceBundle_Base;
class OpenOfficeResourceBundle : public OpenOfficeResourceBundle_Base
{
private:
typedef ::boost::shared_ptr< IResourceType > ResourceTypePtr;
typedef ::std::map< ::rtl::OUString, ResourceTypePtr > ResourceTypes;
::osl::Mutex m_aMutex;
Reference< XResourceBundle > m_xParent;
Locale m_aLocale;
SimpleResMgr* m_pResourceManager;
ResourceTypes m_aResourceTypes;
public:
OpenOfficeResourceBundle(
const Reference< XComponentContext >& _rxContext,
const ::rtl::OUString& _rBaseName,
const Locale& _rLocale
);
protected:
~OpenOfficeResourceBundle();
public:
// XResourceBundle
virtual ::com::sun::star::uno::Reference< ::com::sun::star::resource::XResourceBundle > SAL_CALL getParent() throw (::com::sun::star::uno::RuntimeException);
virtual void SAL_CALL setParent( const ::com::sun::star::uno::Reference< ::com::sun::star::resource::XResourceBundle >& _parent ) throw (::com::sun::star::uno::RuntimeException);
virtual ::com::sun::star::lang::Locale SAL_CALL getLocale( ) throw (::com::sun::star::uno::RuntimeException);
virtual ::com::sun::star::uno::Any SAL_CALL getDirectElement( const ::rtl::OUString& key ) throw (::com::sun::star::uno::RuntimeException);
// XNameAccess (base of XResourceBundle)
virtual ::com::sun::star::uno::Any SAL_CALL getByName( const ::rtl::OUString& aName ) throw (::com::sun::star::container::NoSuchElementException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException);
virtual ::com::sun::star::uno::Sequence< ::rtl::OUString > SAL_CALL getElementNames( ) throw (::com::sun::star::uno::RuntimeException);
virtual ::sal_Bool SAL_CALL hasByName( const ::rtl::OUString& aName ) throw (::com::sun::star::uno::RuntimeException);
// XElementAccess (base of XNameAccess)
virtual ::com::sun::star::uno::Type SAL_CALL getElementType( ) throw (::com::sun::star::uno::RuntimeException);
virtual ::sal_Bool SAL_CALL hasElements( ) throw (::com::sun::star::uno::RuntimeException);
private:
/** retrievs the element with the given key, without asking our parent bundle
@param _key
the key of the element to retrieve
@param _out_Element
will contained the retrieved element upon successful return. If the method is unsuccessful, the
value will not be touched.
@return
<TRUE/> if and only if the element could be retrieved
@precond
our mutex is locked
*/
bool impl_getDirectElement_nothrow( const ::rtl::OUString& _key, Any& _out_Element ) const;
/** retrieves the resource type and id from a given resource key, which assembles those two
@param _key
the resource key as got via a public API call
@param _out_resourceType
the resource type, if successful
@param _out_resourceId
the resource id, if successful
@return
<TRUE/> if and only if the given key specifies a known resource type, and contains a valid
resource id
*/
bool impl_getResourceTypeAndId_nothrow( const ::rtl::OUString& _key, ResourceTypePtr& _out_resourceType, sal_Int32& _out_resourceId ) const;
};
//====================================================================
//= OpenOfficeResourceLoader
//====================================================================
//--------------------------------------------------------------------
OpenOfficeResourceLoader::OpenOfficeResourceLoader( const Reference< XComponentContext >& _rxContext )
:m_xContext( _rxContext )
{
}
//--------------------------------------------------------------------
Sequence< ::rtl::OUString > OpenOfficeResourceLoader::getSupportedServiceNames_static()
{
Sequence< ::rtl::OUString > aServices( 1 );
aServices[ 0 ] = getSingletonName_static();
return aServices;
}
//--------------------------------------------------------------------
::rtl::OUString OpenOfficeResourceLoader::getImplementationName_static()
{
return ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.resource.OpenOfficeResourceLoader" ) );
}
//--------------------------------------------------------------------
::rtl::OUString OpenOfficeResourceLoader::getSingletonName_static()
{
return ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.resource.OfficeResourceLoader" ) );
}
//--------------------------------------------------------------------
Reference< XInterface > OpenOfficeResourceLoader::Create( const Reference< XComponentContext >& _rxContext )
{
return *( new OpenOfficeResourceLoader( _rxContext ) );
}
//--------------------------------------------------------------------
Reference< XResourceBundle > SAL_CALL OpenOfficeResourceLoader::loadBundle_Default( const ::rtl::OUString& _baseName ) throw (MissingResourceException, RuntimeException)
{
return loadBundle( _baseName, Application::GetSettings().GetUILocale() );
}
//--------------------------------------------------------------------
Reference< XResourceBundle > SAL_CALL OpenOfficeResourceLoader::loadBundle( const ::rtl::OUString& _baseName, const Locale& _locale ) throw (MissingResourceException, RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
Reference< XResourceBundle > xBundle;
ResourceBundleDescriptor resourceDescriptor( _baseName, _locale );
ResourceBundleCache::iterator cachePos = m_aBundleCache.find( resourceDescriptor );
if ( cachePos != m_aBundleCache.end() )
xBundle = cachePos->second;
if ( !xBundle.is() )
{ // not in the cache, or already died
xBundle = new OpenOfficeResourceBundle( m_xContext, _baseName, _locale );
m_aBundleCache.insert( ResourceBundleCache::value_type( resourceDescriptor, xBundle ) );
}
return xBundle;
}
//--------------------------------------------------------------------
ComponentInfo getComponentInfo_OpenOfficeResourceLoader()
{
ComponentInfo aInfo;
aInfo.aSupportedServices = OpenOfficeResourceLoader::getSupportedServiceNames_static();
aInfo.sImplementationName = OpenOfficeResourceLoader::getImplementationName_static();
aInfo.sSingletonName = OpenOfficeResourceLoader::getSingletonName_static();
aInfo.pFactory = &OpenOfficeResourceLoader::Create;
return aInfo;
}
//====================================================================
//= OpenOfficeResourceBundle
//====================================================================
//--------------------------------------------------------------------
OpenOfficeResourceBundle::OpenOfficeResourceBundle( const Reference< XComponentContext >& /*_rxContext*/, const ::rtl::OUString& _rBaseName, const Locale& _rLocale )
:m_aLocale( _rLocale )
,m_pResourceManager( NULL )
{
::rtl::OUString sBaseName( _rBaseName );
m_pResourceManager = new SimpleResMgr( sBaseName, m_aLocale );
if ( !m_pResourceManager->IsValid() )
{
delete m_pResourceManager, m_pResourceManager = NULL;
throw MissingResourceException();
}
// supported resource types so far: strings
m_aResourceTypes[ ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "string" ) ) ] =
ResourceTypePtr( new StringResourceAccess );
}
//--------------------------------------------------------------------
OpenOfficeResourceBundle::~OpenOfficeResourceBundle()
{
delete m_pResourceManager;
}
//--------------------------------------------------------------------
Reference< XResourceBundle > SAL_CALL OpenOfficeResourceBundle::getParent() throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
return m_xParent;
}
//--------------------------------------------------------------------
void SAL_CALL OpenOfficeResourceBundle::setParent( const Reference< XResourceBundle >& _parent ) throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
m_xParent = _parent;
}
//--------------------------------------------------------------------
Locale SAL_CALL OpenOfficeResourceBundle::getLocale( ) throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
return m_aLocale;
}
//--------------------------------------------------------------------
bool OpenOfficeResourceBundle::impl_getResourceTypeAndId_nothrow( const ::rtl::OUString& _key, ResourceTypePtr& _out_resourceType, sal_Int32& _out_resourceId ) const
{
sal_Int32 typeSeparatorPos = _key.indexOf( ':' );
if ( typeSeparatorPos == -1 )
// invalid key
return false;
::rtl::OUString resourceType = _key.copy( 0, typeSeparatorPos );
ResourceTypes::const_iterator typePos = m_aResourceTypes.find( resourceType );
if ( typePos == m_aResourceTypes.end() )
// don't know this resource type
return false;
_out_resourceType = typePos->second;
_out_resourceId = _key.copy( typeSeparatorPos + 1 ).toInt32();
return true;
}
//--------------------------------------------------------------------
bool OpenOfficeResourceBundle::impl_getDirectElement_nothrow( const ::rtl::OUString& _key, Any& _out_Element ) const
{
ResourceTypePtr resourceType;
sal_Int32 resourceId( 0 );
if ( !impl_getResourceTypeAndId_nothrow( _key, resourceType, resourceId ) )
return false;
if ( !m_pResourceManager->IsAvailable( resourceType->getResourceType(), resourceId ) )
// no such resource with the given type/id
return false;
_out_Element = resourceType->getResource( *m_pResourceManager, resourceId );
return _out_Element.hasValue();
}
//--------------------------------------------------------------------
Any SAL_CALL OpenOfficeResourceBundle::getDirectElement( const ::rtl::OUString& _key ) throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
Any aElement;
impl_getDirectElement_nothrow( _key, aElement );
return aElement;
}
//--------------------------------------------------------------------
Any SAL_CALL OpenOfficeResourceBundle::getByName( const ::rtl::OUString& _key ) throw (NoSuchElementException, WrappedTargetException, RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
Any aElement;
if ( !impl_getDirectElement_nothrow( _key, aElement ) )
{
if ( m_xParent.is() )
aElement = m_xParent->getByName( _key );
}
if ( !aElement.hasValue() )
throw NoSuchElementException( ::rtl::OUString(), *this );
return aElement;
}
//--------------------------------------------------------------------
Sequence< ::rtl::OUString > SAL_CALL OpenOfficeResourceBundle::getElementNames( ) throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
OSL_ENSURE( false, "OpenOfficeResourceBundle::getElementNames: not implemented!" );
// the (Simple)ResManager does not provide an API to enumerate the resources
return Sequence< ::rtl::OUString >( );
}
//--------------------------------------------------------------------
::sal_Bool SAL_CALL OpenOfficeResourceBundle::hasByName( const ::rtl::OUString& _key ) throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
ResourceTypePtr resourceType;
sal_Int32 resourceId( 0 );
if ( !impl_getResourceTypeAndId_nothrow( _key, resourceType, resourceId ) )
return sal_False;
if ( !m_pResourceManager->IsAvailable( resourceType->getResourceType(), resourceId ) )
return sal_False;
return sal_True;
}
//--------------------------------------------------------------------
Type SAL_CALL OpenOfficeResourceBundle::getElementType( ) throw (RuntimeException)
{
return ::cppu::UnoType< Any >::get();
}
//--------------------------------------------------------------------
::sal_Bool SAL_CALL OpenOfficeResourceBundle::hasElements( ) throw (RuntimeException)
{
::osl::MutexGuard aGuard( m_aMutex );
OSL_ENSURE( false, "OpenOfficeResourceBundle::hasElements: not implemented!" );
// the (Simple)ResManager does not provide an API to enumerate the resources
return ::sal_Bool( );
}
//........................................................................
} // namespace res
//........................................................................
#endif // EXTENSIONS_SOURCE_RESOURCE_OOORESOURCELOADER_CXX