/* -*- 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 #if defined(MACOSX) && HAVE_FEATURE_READONLY_INSTALLSET #define __ASSERT_MACROS_DEFINE_VERSIONS_WITHOUT_UNDERSCORES 0 #include #include #include #endif #include "sal/config.h" #include #include #include #include #include #include #include #include #include "svx/gallery.hxx" #include "gallery.hrc" #include "svx/galmisc.hxx" #include "svx/galtheme.hxx" #include "svx/gallery1.hxx" #include #include #include // - Namespaces - using namespace ::com::sun::star; // - GalleryThemeEntry - static bool FileExists( const INetURLObject &rURL, const rtl::OUString &rExt ) { INetURLObject aURL( rURL ); aURL.setExtension( rExt ); return FileExists( aURL ); } GalleryThemeEntry::GalleryThemeEntry( bool bCreateUniqueURL, const INetURLObject& rBaseURL, const OUString& rName, bool _bReadOnly, bool _bNewFile, sal_uInt32 _nId, bool _bThemeNameFromResource ) : nId ( _nId ), bReadOnly ( _bReadOnly ), bThemeNameFromResource ( _bThemeNameFromResource ) { INetURLObject aURL( rBaseURL ); DBG_ASSERT( aURL.GetProtocol() != INET_PROT_NOT_VALID, "invalid URL" ); if (bCreateUniqueURL) { INetURLObject aBaseNoCase( ImplGetURLIgnoreCase( rBaseURL ) ); aURL = aBaseNoCase; static sal_Int32 nIdx = 0; while( FileExists( aURL, "thm" ) ) { // create new URLs nIdx++; aURL = aBaseNoCase; aURL.setName( aURL.getName() + OUString::number(nIdx)); } } aURL.setExtension( "thm" ); aThmURL = ImplGetURLIgnoreCase( aURL ); aURL.setExtension( "sdg" ); aSdgURL = ImplGetURLIgnoreCase( aURL ); aURL.setExtension( "sdv" ); aSdvURL = ImplGetURLIgnoreCase( aURL ); aURL.setExtension( "str" ); aStrURL = ImplGetURLIgnoreCase( aURL ); SetModified( _bNewFile ); aName = ReadStrFromIni( "name" ); // This is awful - we shouldn't use these resources if we // possibly can avoid them if( aName.isEmpty() && nId && bThemeNameFromResource ) aName = GAL_RESSTR( RID_GALLERYSTR_THEME_START + (sal_uInt16) nId ); if( aName.isEmpty() ) aName = rName; } INetURLObject GalleryThemeEntry::ImplGetURLIgnoreCase( const INetURLObject& rURL ) const { INetURLObject aURL( rURL ); // check original file name if( !FileExists( aURL ) ) { // check upper case file name aURL.setName( aURL.getName().toAsciiUpperCase() ); if(!FileExists( aURL ) ) { // check lower case file name aURL.setName( aURL.getName().toAsciiLowerCase() ); } } return aURL; } void GalleryThemeEntry::SetName( const OUString& rNewName ) { if( aName != rNewName ) { aName = rNewName; SetModified( true ); bThemeNameFromResource = false; } } void GalleryThemeEntry::SetId( sal_uInt32 nNewId, bool bResetThemeName ) { nId = nNewId; SetModified( true ); bThemeNameFromResource = ( nId && bResetThemeName ); } // - GalleryThemeCacheEntry - class GalleryThemeCacheEntry { private: const GalleryThemeEntry* mpThemeEntry; GalleryTheme* mpTheme; public: GalleryThemeCacheEntry( const GalleryThemeEntry* pThemeEntry, GalleryTheme* pTheme ) : mpThemeEntry( pThemeEntry ), mpTheme( pTheme ) {} ~GalleryThemeCacheEntry() { delete mpTheme; } const GalleryThemeEntry* GetThemeEntry() const { return mpThemeEntry; } GalleryTheme* GetTheme() const { return mpTheme; } }; // - Gallery - Gallery::Gallery( const OUString& rMultiPath ) : nReadTextEncoding ( osl_getThreadTextEncoding() ) , bMultiPath ( false ) { ImplLoad( rMultiPath ); } Gallery::~Gallery() { // erase theme list for ( size_t i = 0, n = aThemeList.size(); i < n; ++i ) delete aThemeList[ i ]; aThemeList.clear(); } Gallery* Gallery::GetGalleryInstance() { static Gallery* pGallery = NULL; if( !pGallery ) { ::osl::MutexGuard aGuard( ::osl::Mutex::getGlobalMutex() ); if( !pGallery ) { pGallery = new Gallery( SvtPathOptions().GetGalleryPath() ); } } return pGallery; } void Gallery::ImplLoad( const OUString& rMultiPath ) { const sal_Int32 nTokenCount = comphelper::string::getTokenCount(rMultiPath, ';'); bool bIsReadOnlyDir; bMultiPath = ( nTokenCount > 0 ); INetURLObject aCurURL(SvtPathOptions().GetConfigPath()); ImplLoadSubDirs( aCurURL, bIsReadOnlyDir ); if( !bIsReadOnlyDir ) aUserURL = aCurURL; if( bMultiPath ) { aRelURL = INetURLObject( rMultiPath.getToken(0, ';') ); for( sal_Int32 i = 0; i < nTokenCount; ++i ) { aCurURL = INetURLObject(rMultiPath.getToken(i, ';')); ImplLoadSubDirs( aCurURL, bIsReadOnlyDir ); if( !bIsReadOnlyDir ) aUserURL = aCurURL; } } else aRelURL = INetURLObject( rMultiPath ); DBG_ASSERT( aUserURL.GetProtocol() != INET_PROT_NOT_VALID, "no writable Gallery user directory available" ); DBG_ASSERT( aRelURL.GetProtocol() != INET_PROT_NOT_VALID, "invalid URL" ); } void Gallery::ImplLoadSubDirs( const INetURLObject& rBaseURL, bool& rbDirIsReadOnly ) { rbDirIsReadOnly = false; try { uno::Reference< ucb::XCommandEnvironment > xEnv; ::ucbhelper::Content aCnt( rBaseURL.GetMainURL( INetURLObject::NO_DECODE ), xEnv, comphelper::getProcessComponentContext() ); uno::Sequence< OUString > aProps( 1 ); aProps.getArray()[ 0 ] = "Url"; uno::Reference< sdbc::XResultSet > xResultSet( aCnt.createCursor( aProps, ::ucbhelper::INCLUDE_DOCUMENTS_ONLY ) ); #if defined(MACOSX) && HAVE_FEATURE_READONLY_INSTALLSET if( rBaseURL.GetProtocol() == INET_PROT_FILE ) { const char *appBundle = [[[NSBundle mainBundle] bundlePath] UTF8String]; OUString path = rBaseURL.GetURLPath(); if( path.startsWith( OUString( appBundle, strlen( appBundle ), RTL_TEXTENCODING_UTF8 ) + "/" ) ) rbDirIsReadOnly = true; } #else try { // check readonlyness the very hard way INetURLObject aTestURL( rBaseURL ); OUString aTestFile( "cdefghij.klm" ); aTestURL.Append( aTestFile ); boost::scoped_ptr pTestStm(::utl::UcbStreamHelper::CreateStream( aTestURL.GetMainURL( INetURLObject::NO_DECODE ), STREAM_WRITE )); if( pTestStm ) { pTestStm->WriteInt32( sal_Int32(1) ); if( pTestStm->GetError() ) rbDirIsReadOnly = true; pTestStm.reset(); KillFile( aTestURL ); } else rbDirIsReadOnly = true; } catch( const ucb::ContentCreationException& ) { } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } #endif if( xResultSet.is() ) { uno::Reference< ucb::XContentAccess > xContentAccess( xResultSet, uno::UNO_QUERY ); if( xContentAccess.is() ) { static const char s_sTitle[] = "Title"; static const char s_sIsReadOnly[] = "IsReadOnly"; static const char s_sSDG_EXT[] = "sdg"; static const char s_sSDV_EXT[] = "sdv"; while( xResultSet->next() ) { INetURLObject aThmURL( xContentAccess->queryContentIdentifierString() ); if(aThmURL.GetExtension().equalsIgnoreAsciiCase("thm")) { INetURLObject aSdgURL( aThmURL); aSdgURL.SetExtension( s_sSDG_EXT ); INetURLObject aSdvURL( aThmURL ); aSdvURL.SetExtension( s_sSDV_EXT ); OUString aTitle; bool bReadOnly = false; try { ::ucbhelper::Content aThmCnt( aThmURL.GetMainURL( INetURLObject::NO_DECODE ), xEnv, comphelper::getProcessComponentContext() ); ::ucbhelper::Content aSdgCnt( aSdgURL.GetMainURL( INetURLObject::NO_DECODE ), xEnv, comphelper::getProcessComponentContext() ); ::ucbhelper::Content aSdvCnt( aSdvURL.GetMainURL( INetURLObject::NO_DECODE ), xEnv, comphelper::getProcessComponentContext() ); try { aThmCnt.getPropertyValue( s_sTitle ) >>= aTitle; } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } if( !aTitle.isEmpty() ) { try { aThmCnt.getPropertyValue( s_sIsReadOnly ) >>= bReadOnly; } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } if( !bReadOnly ) { try { aSdgCnt.getPropertyValue( s_sTitle ) >>= aTitle; } catch( const ::com::sun::star::uno::RuntimeException& ) { } catch( const ::com::sun::star::uno::Exception& ) { } if( !aTitle.isEmpty() ) { try { aSdgCnt.getPropertyValue( s_sIsReadOnly ) >>= bReadOnly; } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } } } if( !bReadOnly ) { try { aSdvCnt.getPropertyValue( s_sTitle ) >>= aTitle; } catch( const ::com::sun::star::uno::RuntimeException& ) { } catch( const ::com::sun::star::uno::Exception& ) { } if( !aTitle.isEmpty() ) { try { aSdvCnt.getPropertyValue( s_sIsReadOnly ) >>= bReadOnly; } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } } } GalleryThemeEntry* pEntry = GalleryTheme::CreateThemeEntry( aThmURL, rbDirIsReadOnly || bReadOnly ); if( pEntry ) aThemeList.push_back( pEntry ); } } catch( const ucb::ContentCreationException& ) { } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } } } } } } catch( const ucb::ContentCreationException& ) { } catch( const uno::RuntimeException& ) { } catch( const uno::Exception& ) { } } GalleryThemeEntry* Gallery::ImplGetThemeEntry( const OUString& rThemeName ) { GalleryThemeEntry* pFound = NULL; if( !rThemeName.isEmpty() ) { for ( size_t i = 0, n = aThemeList.size(); i < n && !pFound; ++i ) if( rThemeName == aThemeList[ i ]->GetThemeName() ) pFound = aThemeList[ i ]; } return pFound; } OUString Gallery::GetThemeName( sal_uIntPtr nThemeId ) const { GalleryThemeEntry* pFound = NULL; for ( size_t i = 0, n = aThemeList.size(); i < n && !pFound; ++i ) { GalleryThemeEntry* pEntry = aThemeList[ i ]; if( nThemeId == pEntry->GetId() ) pFound = pEntry; } // try fallback, if no entry was found if( !pFound ) { OUString aFallback; switch( nThemeId ) { case( GALLERY_THEME_3D ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_3D); break; case( GALLERY_THEME_BULLETS ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_BULLETS); break; case( GALLERY_THEME_HOMEPAGE ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_HOMEPAGE); break; case( GALLERY_THEME_POWERPOINT ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_POWERPOINT); break; case( GALLERY_THEME_FONTWORK ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_FONTWORK); break; case( GALLERY_THEME_FONTWORK_VERTICAL ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_FONTWORK_VERTICAL); break; case( GALLERY_THEME_SOUNDS ): aFallback = GAL_RESSTR(RID_GALLERYSTR_THEME_SOUNDS); break; case( RID_GALLERYSTR_THEME_ARROWS ): case( RID_GALLERYSTR_THEME_COMPUTERS ): case( RID_GALLERYSTR_THEME_DIAGRAMS ): case( RID_GALLERYSTR_THEME_EDUCATION ): case( RID_GALLERYSTR_THEME_ENVIRONMENT ): case( RID_GALLERYSTR_THEME_FINANCE ): case( RID_GALLERYSTR_THEME_PEOPLE ): case( RID_GALLERYSTR_THEME_SYMBOLS ): case( RID_GALLERYSTR_THEME_TRANSPORT ): case( RID_GALLERYSTR_THEME_TXTSHAPES ): aFallback = GAL_RESSTR(static_cast(nThemeId)); break; default: break; } pFound = const_cast(this)->ImplGetThemeEntry(aFallback); } return( pFound ? pFound->GetThemeName() : OUString() ); } bool Gallery::HasTheme( const OUString& rThemeName ) { return( ImplGetThemeEntry( rThemeName ) != NULL ); } bool Gallery::CreateTheme( const OUString& rThemeName ) { bool bRet = false; if( !HasTheme( rThemeName ) && ( GetUserURL().GetProtocol() != INET_PROT_NOT_VALID ) ) { INetURLObject aURL( GetUserURL() ); aURL.Append( rThemeName ); GalleryThemeEntry* pNewEntry = new GalleryThemeEntry( true, aURL, rThemeName, false, true, 0, false ); aThemeList.push_back( pNewEntry ); delete( new GalleryTheme( this, pNewEntry ) ); Broadcast( GalleryHint( GALLERY_HINT_THEME_CREATED, rThemeName ) ); bRet = true; } return bRet; } bool Gallery::RenameTheme( const OUString& rOldName, const OUString& rNewName ) { GalleryThemeEntry* pThemeEntry = ImplGetThemeEntry( rOldName ); bool bRet = false; // check if the new theme name is already present if( pThemeEntry && !HasTheme( rNewName ) && !pThemeEntry->IsReadOnly() ) { SfxListener aListener; GalleryTheme* pThm = AcquireTheme( rOldName, aListener ); if( pThm ) { const OUString aOldName( rOldName ); pThemeEntry->SetName( rNewName ); pThm->ImplWrite(); Broadcast( GalleryHint( GALLERY_HINT_THEME_RENAMED, aOldName, pThm->GetName() ) ); ReleaseTheme( pThm, aListener ); bRet = true; } } return bRet; } bool Gallery::RemoveTheme( const OUString& rThemeName ) { GalleryThemeEntry* pThemeEntry = ImplGetThemeEntry( rThemeName ); bool bRet = false; if( pThemeEntry && !pThemeEntry->IsReadOnly() ) { Broadcast( GalleryHint( GALLERY_HINT_CLOSE_THEME, rThemeName ) ); SfxListener aListener; GalleryTheme* pThm = AcquireTheme( rThemeName, aListener ); if( pThm ) { INetURLObject aThmURL( pThm->GetThmURL() ); INetURLObject aSdgURL( pThm->GetSdgURL() ); INetURLObject aSdvURL( pThm->GetSdvURL() ); INetURLObject aStrURL( pThm->GetSdvURL() ); ReleaseTheme( pThm, aListener ); KillFile( aThmURL ); KillFile( aSdgURL ); KillFile( aSdvURL ); KillFile( aStrURL ); } for ( GalleryThemeList::iterator it = aThemeList.begin(); it != aThemeList.end(); ++it ) { if ( pThemeEntry == *it ) { delete pThemeEntry; aThemeList.erase( it ); break; } } Broadcast( GalleryHint( GALLERY_HINT_THEME_REMOVED, rThemeName ) ); bRet = true; } return bRet; } GalleryTheme* Gallery::ImplGetCachedTheme(const GalleryThemeEntry* pThemeEntry) { GalleryTheme* pTheme = NULL; if( pThemeEntry ) { for (GalleryCacheThemeList::const_iterator it = aThemeCache.begin(); it != aThemeCache.end(); ++it) { if (pThemeEntry == (*it)->GetThemeEntry()) { pTheme = (*it)->GetTheme(); break; } } if( !pTheme ) { INetURLObject aURL = pThemeEntry->GetThmURL(); DBG_ASSERT( aURL.GetProtocol() != INET_PROT_NOT_VALID, "invalid URL" ); if( FileExists( aURL ) ) { boost::scoped_ptr pIStm(::utl::UcbStreamHelper::CreateStream( aURL.GetMainURL( INetURLObject::NO_DECODE ), STREAM_READ )); if( pIStm ) { try { pTheme = new GalleryTheme( this, (GalleryThemeEntry*) pThemeEntry ); ReadGalleryTheme( *pIStm, *pTheme ); if( pIStm->GetError() ) { delete pTheme, pTheme = NULL; } } catch (const css::ucb::ContentCreationException&) { } } } if( pTheme ) aThemeCache.push_back( new GalleryThemeCacheEntry( pThemeEntry, pTheme )); } } return pTheme; } void Gallery::ImplDeleteCachedTheme( GalleryTheme* pTheme ) { for (GalleryCacheThemeList::iterator it = aThemeCache.begin(); it != aThemeCache.end(); ++it) { if (pTheme == (*it)->GetTheme()) { delete *it; aThemeCache.erase(it); break; } } } GalleryTheme* Gallery::AcquireTheme( const OUString& rThemeName, SfxListener& rListener ) { GalleryTheme* pTheme = NULL; GalleryThemeEntry* pThemeEntry = ImplGetThemeEntry( rThemeName ); if( pThemeEntry && ( ( pTheme = ImplGetCachedTheme( pThemeEntry ) ) != NULL ) ) rListener.StartListening( *pTheme ); return pTheme; } void Gallery::ReleaseTheme( GalleryTheme* pTheme, SfxListener& rListener ) { if( pTheme ) { rListener.EndListening( *pTheme ); if( !pTheme->HasListeners() ) ImplDeleteCachedTheme( pTheme ); } } bool GalleryThemeEntry::IsDefault() const { return ( nId > 0 ) && ( nId != ( RID_GALLERYSTR_THEME_MYTHEME - RID_GALLERYSTR_THEME_START ) ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */