1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_extensions.hxx" 26 27 #ifndef EXTENSIONS_SOURCE_RESOURCE_OOORESOURCELOADER_CXX 28 #define EXTENSIONS_SOURCE_RESOURCE_OOORESOURCELOADER_CXX 29 #include "res_services.hxx" 30 31 /** === begin UNO includes === **/ 32 #include <com/sun/star/resource/XResourceBundleLoader.hpp> 33 #include <com/sun/star/lang/XMultiServiceFactory.hpp> 34 #include <com/sun/star/uno/XComponentContext.hpp> 35 /** === end UNO includes === **/ 36 #include <vcl/svapp.hxx> 37 #include <tools/simplerm.hxx> 38 #include <tools/rcid.h> 39 #include <cppuhelper/implbase1.hxx> 40 #include <cppuhelper/weakref.hxx> 41 42 #include <boost/shared_ptr.hpp> 43 #include <map> 44 45 //........................................................................ 46 namespace res 47 { 48 //........................................................................ 49 50 /** === begin UNO using === **/ 51 using ::com::sun::star::uno::Reference; 52 using ::com::sun::star::resource::XResourceBundleLoader; 53 using ::com::sun::star::resource::XResourceBundle; 54 using ::com::sun::star::resource::MissingResourceException; 55 using ::com::sun::star::uno::XComponentContext; 56 using ::com::sun::star::uno::Sequence; 57 using ::com::sun::star::uno::XInterface; 58 using ::com::sun::star::uno::RuntimeException; 59 using ::com::sun::star::lang::Locale; 60 using ::com::sun::star::uno::Any; 61 using ::com::sun::star::container::NoSuchElementException; 62 using ::com::sun::star::lang::WrappedTargetException; 63 using ::com::sun::star::uno::Type; 64 using ::com::sun::star::uno::WeakReference; 65 /** === end UNO using === **/ 66 67 //==================================================================== 68 //= helper 69 //==================================================================== 70 typedef ::std::pair< ::rtl::OUString, Locale > ResourceBundleDescriptor; 71 72 struct ResourceBundleDescriptorLess : public ::std::binary_function< ResourceBundleDescriptor, ResourceBundleDescriptor, bool > 73 { 74 bool operator()( const ResourceBundleDescriptor& _lhs, const ResourceBundleDescriptor& _rhs ) const 75 { 76 if ( _lhs.first < _rhs.first ) 77 return true; 78 if ( _lhs.second.Language < _rhs.second.Language ) 79 return true; 80 if ( _lhs.second.Country < _rhs.second.Country ) 81 return true; 82 if ( _lhs.second.Variant < _rhs.second.Variant ) 83 return true; 84 return false; 85 } 86 }; 87 88 //==================================================================== 89 //= OpenOfficeResourceLoader 90 //==================================================================== 91 typedef ::cppu::WeakImplHelper1 < XResourceBundleLoader 92 > OpenOfficeResourceLoader_Base; 93 class OpenOfficeResourceLoader : public OpenOfficeResourceLoader_Base 94 { 95 private: 96 typedef ::std::map< ResourceBundleDescriptor, WeakReference< XResourceBundle >, ResourceBundleDescriptorLess > 97 ResourceBundleCache; 98 99 private: 100 Reference< XComponentContext > m_xContext; 101 ::osl::Mutex m_aMutex; 102 ResourceBundleCache m_aBundleCache; 103 104 protected: 105 OpenOfficeResourceLoader( const Reference< XComponentContext >& _rxContext ); 106 107 public: 108 static Sequence< ::rtl::OUString > getSupportedServiceNames_static(); 109 static ::rtl::OUString getImplementationName_static(); 110 static ::rtl::OUString getSingletonName_static(); 111 static Reference< XInterface > Create( const Reference< XComponentContext >& _rxContext ); 112 113 // XResourceBundleLoader 114 virtual Reference< XResourceBundle > SAL_CALL loadBundle_Default( const ::rtl::OUString& aBaseName ) throw (MissingResourceException, RuntimeException); 115 virtual Reference< XResourceBundle > SAL_CALL loadBundle( const ::rtl::OUString& abaseName, const Locale& aLocale ) throw (MissingResourceException, RuntimeException); 116 117 private: 118 OpenOfficeResourceLoader(); // never implemented 119 OpenOfficeResourceLoader( const OpenOfficeResourceLoader& ); // never implemented 120 OpenOfficeResourceLoader& operator=( const OpenOfficeResourceLoader& ); // never implemented 121 }; 122 123 //==================================================================== 124 //= IResourceType 125 //==================================================================== 126 /** encapsulates access to a fixed resource type 127 */ 128 class IResourceType 129 { 130 public: 131 /** returns the RESOURCE_TYPE associated with this instance 132 */ 133 virtual RESOURCE_TYPE getResourceType() const = 0; 134 135 /** reads a single resource from the given resource manager 136 @param _resourceManager 137 the resource manager to read from 138 @param _resourceId 139 the id of the resource to read 140 @return 141 the required resource 142 @precond 143 the caler checked via <code>_resourceManager.IsAvailable( getResourceType(), _resourceId )</code> 144 that the required resource really exists 145 */ 146 virtual Any getResource( SimpleResMgr& _resourceManager, sal_Int32 _resourceId ) const = 0; 147 148 virtual ~IResourceType() { }; 149 }; 150 151 //==================================================================== 152 //= StringResourceAccess 153 //==================================================================== 154 class StringResourceAccess : public IResourceType 155 { 156 public: 157 StringResourceAccess(); 158 159 // IResourceType 160 virtual RESOURCE_TYPE getResourceType() const; 161 virtual Any getResource( SimpleResMgr& _resourceManager, sal_Int32 _resourceId ) const; 162 }; 163 164 //-------------------------------------------------------------------- 165 StringResourceAccess::StringResourceAccess() 166 { 167 } 168 169 //-------------------------------------------------------------------- 170 RESOURCE_TYPE StringResourceAccess::getResourceType() const 171 { 172 return RSC_STRING; 173 } 174 175 //-------------------------------------------------------------------- 176 Any StringResourceAccess::getResource( SimpleResMgr& _resourceManager, sal_Int32 _resourceId ) const 177 { 178 OSL_PRECOND( _resourceManager.IsAvailable( getResourceType(), _resourceId ), "StringResourceAccess::getResource: precondition not met!" ); 179 Any aResource; 180 aResource <<= ::rtl::OUString( _resourceManager.ReadString( _resourceId ) ); 181 return aResource; 182 } 183 184 //==================================================================== 185 //= OpenOfficeResourceBundle 186 //==================================================================== 187 typedef ::cppu::WeakImplHelper1 < XResourceBundle 188 > OpenOfficeResourceBundle_Base; 189 class OpenOfficeResourceBundle : public OpenOfficeResourceBundle_Base 190 { 191 private: 192 typedef ::boost::shared_ptr< IResourceType > ResourceTypePtr; 193 typedef ::std::map< ::rtl::OUString, ResourceTypePtr > ResourceTypes; 194 195 ::osl::Mutex m_aMutex; 196 Reference< XResourceBundle > m_xParent; 197 Locale m_aLocale; 198 SimpleResMgr* m_pResourceManager; 199 ResourceTypes m_aResourceTypes; 200 201 public: 202 OpenOfficeResourceBundle( 203 const Reference< XComponentContext >& _rxContext, 204 const ::rtl::OUString& _rBaseName, 205 const Locale& _rLocale 206 ); 207 208 protected: 209 ~OpenOfficeResourceBundle(); 210 211 public: 212 // XResourceBundle 213 virtual ::com::sun::star::uno::Reference< ::com::sun::star::resource::XResourceBundle > SAL_CALL getParent() throw (::com::sun::star::uno::RuntimeException); 214 virtual void SAL_CALL setParent( const ::com::sun::star::uno::Reference< ::com::sun::star::resource::XResourceBundle >& _parent ) throw (::com::sun::star::uno::RuntimeException); 215 virtual ::com::sun::star::lang::Locale SAL_CALL getLocale( ) throw (::com::sun::star::uno::RuntimeException); 216 virtual ::com::sun::star::uno::Any SAL_CALL getDirectElement( const ::rtl::OUString& key ) throw (::com::sun::star::uno::RuntimeException); 217 218 // XNameAccess (base of XResourceBundle) 219 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); 220 virtual ::com::sun::star::uno::Sequence< ::rtl::OUString > SAL_CALL getElementNames( ) throw (::com::sun::star::uno::RuntimeException); 221 virtual ::sal_Bool SAL_CALL hasByName( const ::rtl::OUString& aName ) throw (::com::sun::star::uno::RuntimeException); 222 223 // XElementAccess (base of XNameAccess) 224 virtual ::com::sun::star::uno::Type SAL_CALL getElementType( ) throw (::com::sun::star::uno::RuntimeException); 225 virtual ::sal_Bool SAL_CALL hasElements( ) throw (::com::sun::star::uno::RuntimeException); 226 227 private: 228 /** retrievs the element with the given key, without asking our parent bundle 229 @param _key 230 the key of the element to retrieve 231 @param _out_Element 232 will contained the retrieved element upon successful return. If the method is unsuccessful, the 233 value will not be touched. 234 @return 235 <TRUE/> if and only if the element could be retrieved 236 @precond 237 our mutex is locked 238 */ 239 bool impl_getDirectElement_nothrow( const ::rtl::OUString& _key, Any& _out_Element ) const; 240 241 /** retrieves the resource type and id from a given resource key, which assembles those two 242 @param _key 243 the resource key as got via a public API call 244 @param _out_resourceType 245 the resource type, if successful 246 @param _out_resourceId 247 the resource id, if successful 248 @return 249 <TRUE/> if and only if the given key specifies a known resource type, and contains a valid 250 resource id 251 */ 252 bool impl_getResourceTypeAndId_nothrow( const ::rtl::OUString& _key, ResourceTypePtr& _out_resourceType, sal_Int32& _out_resourceId ) const; 253 }; 254 255 //==================================================================== 256 //= OpenOfficeResourceLoader 257 //==================================================================== 258 //-------------------------------------------------------------------- 259 OpenOfficeResourceLoader::OpenOfficeResourceLoader( const Reference< XComponentContext >& _rxContext ) 260 :m_xContext( _rxContext ) 261 { 262 } 263 264 //-------------------------------------------------------------------- 265 Sequence< ::rtl::OUString > OpenOfficeResourceLoader::getSupportedServiceNames_static() 266 { 267 Sequence< ::rtl::OUString > aServices( 1 ); 268 aServices[ 0 ] = getSingletonName_static(); 269 return aServices; 270 } 271 272 //-------------------------------------------------------------------- 273 ::rtl::OUString OpenOfficeResourceLoader::getImplementationName_static() 274 { 275 return ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.comp.resource.OpenOfficeResourceLoader" ) ); 276 } 277 278 //-------------------------------------------------------------------- 279 ::rtl::OUString OpenOfficeResourceLoader::getSingletonName_static() 280 { 281 return ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.resource.OfficeResourceLoader" ) ); 282 } 283 284 //-------------------------------------------------------------------- 285 Reference< XInterface > OpenOfficeResourceLoader::Create( const Reference< XComponentContext >& _rxContext ) 286 { 287 return *( new OpenOfficeResourceLoader( _rxContext ) ); 288 } 289 290 //-------------------------------------------------------------------- 291 Reference< XResourceBundle > SAL_CALL OpenOfficeResourceLoader::loadBundle_Default( const ::rtl::OUString& _baseName ) throw (MissingResourceException, RuntimeException) 292 { 293 return loadBundle( _baseName, Application::GetSettings().GetUILocale() ); 294 } 295 296 //-------------------------------------------------------------------- 297 Reference< XResourceBundle > SAL_CALL OpenOfficeResourceLoader::loadBundle( const ::rtl::OUString& _baseName, const Locale& _locale ) throw (MissingResourceException, RuntimeException) 298 { 299 ::osl::MutexGuard aGuard( m_aMutex ); 300 301 Reference< XResourceBundle > xBundle; 302 303 ResourceBundleDescriptor resourceDescriptor( _baseName, _locale ); 304 ResourceBundleCache::iterator cachePos = m_aBundleCache.find( resourceDescriptor ); 305 if ( cachePos != m_aBundleCache.end() ) 306 xBundle = cachePos->second; 307 308 if ( !xBundle.is() ) 309 { // not in the cache, or already died 310 xBundle = new OpenOfficeResourceBundle( m_xContext, _baseName, _locale ); 311 m_aBundleCache.insert( ResourceBundleCache::value_type( resourceDescriptor, xBundle ) ); 312 } 313 314 return xBundle; 315 } 316 317 //-------------------------------------------------------------------- 318 ComponentInfo getComponentInfo_OpenOfficeResourceLoader() 319 { 320 ComponentInfo aInfo; 321 aInfo.aSupportedServices = OpenOfficeResourceLoader::getSupportedServiceNames_static(); 322 aInfo.sImplementationName = OpenOfficeResourceLoader::getImplementationName_static(); 323 aInfo.sSingletonName = OpenOfficeResourceLoader::getSingletonName_static(); 324 aInfo.pFactory = &OpenOfficeResourceLoader::Create; 325 return aInfo; 326 } 327 328 //==================================================================== 329 //= OpenOfficeResourceBundle 330 //==================================================================== 331 //-------------------------------------------------------------------- 332 OpenOfficeResourceBundle::OpenOfficeResourceBundle( const Reference< XComponentContext >& /*_rxContext*/, const ::rtl::OUString& _rBaseName, const Locale& _rLocale ) 333 :m_aLocale( _rLocale ) 334 ,m_pResourceManager( NULL ) 335 { 336 ::rtl::OUString sBaseName( _rBaseName ); 337 m_pResourceManager = new SimpleResMgr( sBaseName, m_aLocale ); 338 339 if ( !m_pResourceManager->IsValid() ) 340 { 341 delete m_pResourceManager, m_pResourceManager = NULL; 342 throw MissingResourceException(); 343 } 344 345 // supported resource types so far: strings 346 m_aResourceTypes[ ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "string" ) ) ] = 347 ResourceTypePtr( new StringResourceAccess ); 348 } 349 350 //-------------------------------------------------------------------- 351 OpenOfficeResourceBundle::~OpenOfficeResourceBundle() 352 { 353 delete m_pResourceManager; 354 } 355 356 //-------------------------------------------------------------------- 357 Reference< XResourceBundle > SAL_CALL OpenOfficeResourceBundle::getParent() throw (RuntimeException) 358 { 359 ::osl::MutexGuard aGuard( m_aMutex ); 360 return m_xParent; 361 } 362 363 //-------------------------------------------------------------------- 364 void SAL_CALL OpenOfficeResourceBundle::setParent( const Reference< XResourceBundle >& _parent ) throw (RuntimeException) 365 { 366 ::osl::MutexGuard aGuard( m_aMutex ); 367 m_xParent = _parent; 368 } 369 370 //-------------------------------------------------------------------- 371 Locale SAL_CALL OpenOfficeResourceBundle::getLocale( ) throw (RuntimeException) 372 { 373 ::osl::MutexGuard aGuard( m_aMutex ); 374 return m_aLocale; 375 } 376 377 //-------------------------------------------------------------------- 378 bool OpenOfficeResourceBundle::impl_getResourceTypeAndId_nothrow( const ::rtl::OUString& _key, ResourceTypePtr& _out_resourceType, sal_Int32& _out_resourceId ) const 379 { 380 sal_Int32 typeSeparatorPos = _key.indexOf( ':' ); 381 if ( typeSeparatorPos == -1 ) 382 // invalid key 383 return false; 384 385 ::rtl::OUString resourceType = _key.copy( 0, typeSeparatorPos ); 386 387 ResourceTypes::const_iterator typePos = m_aResourceTypes.find( resourceType ); 388 if ( typePos == m_aResourceTypes.end() ) 389 // don't know this resource type 390 return false; 391 392 _out_resourceType = typePos->second; 393 _out_resourceId = _key.copy( typeSeparatorPos + 1 ).toInt32(); 394 return true; 395 } 396 397 //-------------------------------------------------------------------- 398 bool OpenOfficeResourceBundle::impl_getDirectElement_nothrow( const ::rtl::OUString& _key, Any& _out_Element ) const 399 { 400 ResourceTypePtr resourceType; 401 sal_Int32 resourceId( 0 ); 402 if ( !impl_getResourceTypeAndId_nothrow( _key, resourceType, resourceId ) ) 403 return false; 404 405 if ( !m_pResourceManager->IsAvailable( resourceType->getResourceType(), resourceId ) ) 406 // no such resource with the given type/id 407 return false; 408 409 _out_Element = resourceType->getResource( *m_pResourceManager, resourceId ); 410 return _out_Element.hasValue(); 411 } 412 413 //-------------------------------------------------------------------- 414 Any SAL_CALL OpenOfficeResourceBundle::getDirectElement( const ::rtl::OUString& _key ) throw (RuntimeException) 415 { 416 ::osl::MutexGuard aGuard( m_aMutex ); 417 418 Any aElement; 419 impl_getDirectElement_nothrow( _key, aElement ); 420 return aElement; 421 } 422 423 //-------------------------------------------------------------------- 424 Any SAL_CALL OpenOfficeResourceBundle::getByName( const ::rtl::OUString& _key ) throw (NoSuchElementException, WrappedTargetException, RuntimeException) 425 { 426 ::osl::MutexGuard aGuard( m_aMutex ); 427 428 Any aElement; 429 if ( !impl_getDirectElement_nothrow( _key, aElement ) ) 430 { 431 if ( m_xParent.is() ) 432 aElement = m_xParent->getByName( _key ); 433 } 434 435 if ( !aElement.hasValue() ) 436 throw NoSuchElementException( ::rtl::OUString(), *this ); 437 438 return aElement; 439 } 440 441 //-------------------------------------------------------------------- 442 Sequence< ::rtl::OUString > SAL_CALL OpenOfficeResourceBundle::getElementNames( ) throw (RuntimeException) 443 { 444 ::osl::MutexGuard aGuard( m_aMutex ); 445 OSL_ENSURE( false, "OpenOfficeResourceBundle::getElementNames: not implemented!" ); 446 // the (Simple)ResManager does not provide an API to enumerate the resources 447 return Sequence< ::rtl::OUString >( ); 448 } 449 450 //-------------------------------------------------------------------- 451 ::sal_Bool SAL_CALL OpenOfficeResourceBundle::hasByName( const ::rtl::OUString& _key ) throw (RuntimeException) 452 { 453 ::osl::MutexGuard aGuard( m_aMutex ); 454 455 ResourceTypePtr resourceType; 456 sal_Int32 resourceId( 0 ); 457 if ( !impl_getResourceTypeAndId_nothrow( _key, resourceType, resourceId ) ) 458 return sal_False; 459 460 if ( !m_pResourceManager->IsAvailable( resourceType->getResourceType(), resourceId ) ) 461 return sal_False; 462 463 return sal_True; 464 } 465 466 //-------------------------------------------------------------------- 467 Type SAL_CALL OpenOfficeResourceBundle::getElementType( ) throw (RuntimeException) 468 { 469 return ::cppu::UnoType< Any >::get(); 470 } 471 472 //-------------------------------------------------------------------- 473 ::sal_Bool SAL_CALL OpenOfficeResourceBundle::hasElements( ) throw (RuntimeException) 474 { 475 ::osl::MutexGuard aGuard( m_aMutex ); 476 OSL_ENSURE( false, "OpenOfficeResourceBundle::hasElements: not implemented!" ); 477 // the (Simple)ResManager does not provide an API to enumerate the resources 478 return ::sal_Bool( ); 479 } 480 481 //........................................................................ 482 } // namespace res 483 //........................................................................ 484 485 #endif // EXTENSIONS_SOURCE_RESOURCE_OOORESOURCELOADER_CXX 486